Android Crash 案例分析

四、ActivityNotFoundException Activity未找到异常

五、IllegalStateException 非法状态异常

六、ArrayIndexOutOfBoundsException 数组越界异常

七、SecurityException 安全异常

八、llegalArgumentException: Service not registered 服务未注册异常

九、BadTokenException:

Exception 在Android 中经常会遇到,那么遇到异常我们该如何解决,本文将举例解决部分Android看法中遇到的异常。

一、NullPointerException 空指针

==========================

NullPointerException在开发中经常会碰到,比如引用的对象为空,数组为空等等都会引起空指针异常,如不及时处理,就会导致 应用Crash

1. 数组 NullPointerException


不能向一个null数组元素赋值,获取长度,否则报

NullPointerException: Attempt to write to null array

NullPointerException Attempt to get length of null array,以下代码会引起上面两种空指针异常。

2. 数组NullPointerException 代码举例



    `public static void ArrayNullPointer() {           /**            * 数组空指针 NullPointerException            *            * 1.获取null数组长度            * 2.为null 数组元素复制            * */           int[] array = null;           // 1. NullPointerException: Attempt to get length of null array           int length = array.length;           // 2. NullPointerException: Attempt to write to null array           array[0] = 1;          }`

NullPointerException 代码举例

3. 数组NullPointerException Log 举例


  • Log 信息如下

获取 空数组长度导致的 NullPointerException 如下:


12-27 17:17:44.627  8839  8839 E AndroidRuntime:  Caused by: java.lang.NullPointerException:                                                       Attempt to get length of null array   12-27 17:17:44.627  8839  8839 E AndroidRuntime:    at com.programandroid.Exception.NullPointerException.ArrayNullPointer                                                      //产生空指针代码行                                                      (NullPointerException.java:32)   

4. Log 分析如下


数组NullPointerException

空数组无法获取下标内容,如果获取则会导致NullPointerException


12-27 17:23:24.168 11649 11649 E AndroidRuntime: Caused by: java.lang.NullPointerException: Attempt to write to null array   12-27 17:23:24.168 11649 11649 E AndroidRuntime:    at com.programandroid.Exception.NullPointerException.ArrayNullPointer(NullPointerException.java:34)   12-27 17:23:24.168 11649 11649 E AndroidRuntime:    at com.programandroid.Exception.ExceptionActivity.NullPointerException(ExceptionActivity.java:37)   

5. Object 对象 NullPointerException


对象空指针,这个是常见的空指针,主要是因为引用一个null 对象,进而导致空指针,常报以下错误

Attempt to invoke a virtual method on a null object reference,以下代码可能会引起空指针异常。

6. object 对象 NullPointerException 代码举例


简单代码举例如下:


    `public static void ListNullPointer() {              ArrayList<String> mArrayList = null;               mArrayList.size();       }`

Object 对象 NullPointerException

7. object 对象 NullPointerException log 举例


  • Log 信息如下:

12-27 17:28:22.565 12725 12725 E AndroidRuntime: Caused by: java.lang.NullPointerException: Attempt to invoke a virtual method on a null object reference   12-27 17:28:22.565 12725 12725 E AndroidRuntime:    at com.programandroid.Exception.NullPointerException.ListNullPointer(NullPointerException.java:45)   12-27 17:28:22.565 12725 12725 E AndroidRuntime:    at com.programandroid.Exception.ExceptionActivity.NullPointerException(ExceptionActivity.java:37)   

8. object 对象 NullPointerException Log 分析如下:


Object NullPointerException

9. NullPointerException 解决方案


  • 1.使用时多注意判断对象是否为空

  • 规避空指针举例如下:


    `public static void ListNullPointer() {              ArrayList<String> mArrayList = null;           if (mArrayList != null) {               mArrayList.size();           }       }`

使用对象是,最好判断对象是否为空

  • 2.使用try-catch将抛出的异常抓住

try-catch 可以抓住抛出的异常,使应用程序不崩溃,但是,这个不是从根本上解决问题,会引起一些莫名其妙的问题。


    `public static void ListNullPointer() {               try {                   ArrayList<String> mArrayList = null;                   mArrayList.size();               } catch (Exception e) {                   // TODO: handle exception               }       }`

try-catch 代码异常,防止app crash

二、 ClassCastException 类型转换异常

============================

ClassCastException 类型转换异常:

此异常发生在类型转换时,并且在编译期间,编译器不会提示报错,但是当运行时,如果存在此异常,可能会导致app崩溃 crash

比如当父类强制转换为子类时,ClassCastException 就会发生

1. 以下代码 会引起 ClassCastException


请勿 父类强制转换为子类,否则就会发生ClassCastException异常。


public void ClassCastExample() {           Fruit banana = new Fruit();           /**            * ClassCastException            *            * 1. 此处强制转换,会导致 app 编译没问题,运行挂掉, Caused by:            * java.lang.ClassCastException:            * com.programandroid.Exception.ExceptionActivity$ Fruit cannot be cast            * to com.programandroid.Exception.ExceptionActivity$Apple            *            ***/           Apple apple = (Apple) banana;          }          /**        * ClassCastException        *        * 2. 此处强转回导致app crash return (Apple) banana;        * */       public Apple isRight() {           Fruit banana = new Fruit();           return (Apple) banana;       }          class Fruit {           public Fruit() {           }       }          class Apple extends Fruit {           public Apple() {           }       }   

ClassCastException 类型转换异常举例

2. ClassCastException Log 举例


ClassCastException通常会打印以下类似信息


Caused by: java.lang.ClassCastException:   com.programandroid.Exception.ExceptionActivity$   Fruit cannot be cast to com.programandroid.Exception.ExceptionActivity$Apple   

3. ClassCastException Log 分析


ClassCastException log 分析

4. ClassCastException 解决方案


使用try-catch抓住异常,或者从代码上解决根本问题。

使用 try-catch抓住 ClassCastException异常

5. Android 手机 Settings ClassCastException 解决方案


举例是为了更好的解决开发中的异常。比如在开发中,使用 monkey 测试Settings模块时,报出的 ClassCastExceptionSettings代码比较多,一时也无法看完,此时,try-catch 也是一种不错的选择。

比如monkey测试某平台代码时,报出以下异常

  • log 信息如下:

FATAL EXCEPTION: ApplicationsState.Loader   01-05 03:36:56.101  6304  6941 E AndroidRuntime: Process: com.android.settings, PID: 6304   01-05 03:36:56.101  6304  6941 E AndroidRuntime: java.lang.ClassCastException:                                                      com.android.settings.datausage.AppStateDataUsageBridge$DataUsageState                                                      cannot be cast to com.android.settings.notification.NotificationBackend$AppRow                                                         01-05 03:36:56.101  6304  6941 E AndroidRuntime:    at com.android.settings.applications.AppStateNotificationBridge$3.filterApp(AppStateNotificationBridge.java:110)   

6. Settings ClassCastException Log分析


Settings ClassCastException Log1

Settings ClassCastException Log2

7. Setting crash ClassCastException 解决方案:


try-catch 异常报错的地方

try-catch 异常报错的地方

try-catch 异常报错的地方

三、IndexOutOfBoundsException 下标越界异常

==================================

List 在开发中经常会被用的,那么错误的使用下标,将会导致IndexOutOfBoundsException 越界异常。以下代码就会引起IndexOutOfBoundsException异常

1. IndexOutOfBoundsException 代码举例


IndexOutOfBoundsException 异常举例

2. IndexOutOfBoundsException Log举例


  • Log 信息如下:

12-27 17:41:24.231 16891 16891 E AndroidRuntime: Caused by: java.lang.IndexOutOfBoundsException: Index: 0, Size: 0   12-27 17:41:24.231 16891 16891 E AndroidRuntime:    at java.util.ArrayList.get(ArrayList.java:411)   12-27 17:41:24.231 16891 16891 E AndroidRuntime:    at com.programandroid.Exception.IndexOutOfBoundsException.isAppOnRecent(IndexOutOfBoundsException.java:40)   12-27 17:41:24.231 16891 16891 E AndroidRuntime:    at com.programandroid.Exception.ExceptionActivity.IndexOutOfBoundsException(ExceptionActivity.java:80)   

3. Log 分析如下:


IndexOutOfBoundsException Log分析

4. IndexOutOfBoundsException 解决方案


在使用时判断对象内容是否为0.

使用判断List 的size是否为0

四、ActivityNotFoundException

===========================

ActivityNotFoundException 常见于Eclipse 开发Android中,Android studio 已经帮忙自动生成Activity,以及布局文件。

主要原因是未在AndroidMainfest.xml文件中注册,如未注册,会引起app crash ,crash log如下:

ActivityNotFoundException: Unable to find explicit activity class

1. ActivityNotFoundException 代码举例


比如以下代码会引起此异常

Activity未在Androidmainfest.xml 中注册会引起ActivityNotFoundException

2. ActivityNotFoundException Log 举例


  • Log信息如下:

12-27 17:46:05.994 17893 17893 E AndroidRuntime: Caused by: android.content.ActivityNotFoundException: Unable to find explicit activity class {com.programandroid/com.programandroid.Test.TestActivity}; have you declared this activity in your AndroidManifest.xml?   12-27 17:46:05.994 17893 17893 E AndroidRuntime:    at android.app.Instrumentation.checkStartActivityResult(Instrumentation.java:1810)   

3. Log 分析如下:


ActivityNotFoundException Log分析

4. ActivityNotFoundException 解决方案


AndroidMainfest.xml中注册即可

四大组件一定,一定要在AndroidMainfest.xml 中注册

五、IllegalStateException

=======================

IllegalStateException 非法状态异常,是因为软件中代码状态非法导致的。

以下代码会引起IllegalStateException 。当Button控件声明android:onClick="IllegalStateException" 却未在Java代码中使用时,点击Button,就会出现此类异常。

1. IllegalStateException 代码举例


IllegalStateException 代码举例

2. IllegalStateException Log 举例


  • log信息如下:

12-27 16:07:41.158  1715  1715 E AndroidRuntime: FATAL EXCEPTION: main   12-27 16:07:41.158  1715  1715 E AndroidRuntime: Process: com.programandroid, PID: 1715   12-27 16:07:41.158  1715  1715 E AndroidRuntime: java.lang.IllegalStateException:                                                    Could not find method IllegalStateException(View) in a parent                                                   or ancestor Context for android:onClick attribute defined on view class                                                    android.widget.Button with id 'btn_on_click'   12-27 16:07:41.158  1715  1715 E AndroidRuntime:    at android.view.View$DeclaredOnClickListener.resolveMethod(View.java:4781)   12-27 16:07:41.158  1715  1715 E AndroidRuntime:    at android.view.View$DeclaredOnClickListener.onClick(View.java:4740)   

3. IllegalStateException Log分析如下:


IllegalStateException Log截图

4. IllegalStateException 解决方案


IllegalStateException 类异常很多,不同的代码会有不同的解决方案,上述举例解决方案如下

IllegalStateException

六、 ArrayIndexOutOfBoundsException 数组越界异常

========================================

数组在代码中经常被用到,当适用数组下标不当时,就会出现ArrayIndexOutOfBoundsException。比如数组长度为4,但你要引用下标为5的元素,这时候,就会异常crash

1. ArrayIndexOutOfBoundsException 代码举例:



    `public static void ArrayIndexOutOfBounds() {              String[] mStrings = { "a", "b", "c", "d" };           String testsString = mStrings[5];       }`

ArrayIndexOutOfBoundsException 代码举例

2. ArrayIndexOutOfBoundsException Log举例:


  • Log信息如下:

12-27 17:51:15.420 19185 19185 E AndroidRuntime: Caused by: java.lang.ArrayIndexOutOfBoundsException: length=4; index=5   12-27 17:51:15.420 19185 19185 E AndroidRuntime:    at com.programandroid.Exception.ArrayIndexOutOfBoundsException.ArrayIndexOutOfBounds(ArrayIndexOutOfBoundsException.java:20)   12-27 17:51:15.420 19185 19185 E AndroidRuntime:    at com.programandroid.Exception.ExceptionActivity.ArrayIndexOutOfBoundsException(ExceptionActivity.java:105)   12-27 17:51:15.420 19185 19185 E AndroidRuntime:    ... 11 more   

3. ArrayIndexOutOfBoundsException Log分析如下:


ArrayIndexOutOfBoundsException Log分析

4. ArrayIndexOutOfBoundsException解决方案


  • 1.正确使用数组下标

  • 2.如果不确定数组长度,请先获取长度,然后在判断下标是否大于等于数组长度。

  • 3.try-catch 抓住异常,防止crash,但不能从根本上解决问题。

七、SecurityException 安全异常

========================

SecurityException 安全异常在Android 中也会经常发生,主要是Android 的安全机制原因造成的,为了管理应用获取手机的一些敏感信息,Android安全机制规定,必须在AndroidMainfest.xml文件中声明,并且,Android 6.0之后,获取手机敏感信息时候,需要动态申请权限,只有用户授权后才可以获取手机敏感信息。

1. SecurityException 代码举例


获取手机的IMEI 号属于手机的敏感信息


/**        *        * <!-- 读取手机IMEI的设备权限 -->        *        * <uses-permission android:name="android.permission.READ_PHONE_STATE" />        * */       public static String getIMEI(Context context) {           TelephonyManager tm = (TelephonyManager) context                   .getSystemService(Context.TELEPHONY_SERVICE);           String deviceId = tm.getDeviceId();           if (deviceId == null) {               return "UnKnown";           } else {               return deviceId;           }       }   

获取手机IMEI号

2. SecurityException log举例



12-27 18:05:55.663 21467 21467 E AndroidRuntime: Caused by: java.lang.SecurityException: getDeviceId: Neither user 10117 nor current process has android.permission.READ_PHONE_STATE.   12-27 18:05:55.663 21467 21467 E AndroidRuntime:    at android.os.Parcel.readException(Parcel.java:1683)   12-27 18:05:55.663 21467 21467 E AndroidRuntime:    at android.os.Parcel.readException(Parcel.java:1636)   12-27 18:05:55.663 21467 21467 E AndroidRuntime:    at com.android.internal.telephony.ITelephony$Stub$Proxy.getDeviceId(ITelephony.java:4281)   

以下是一些分析 Android 崩溃的步骤案例: 步骤一:获取崩溃日志 获取崩溃日志是分析 Android 崩溃的第一步。您可以在开发者选项中启用 USB 调试,并使用 ADB 工具连接到设备,然后使用以下命令获取崩溃日志: ``` adb logcat -d > crash_log.txt ``` 步骤二:查看崩溃日志 崩溃日志包含有关崩溃的详细信息,例如异常类型、堆栈跟踪和线程信息。您可以使用文本编辑器或日志查看器查看崩溃日志并查找关键信息。以下是一个示例崩溃日志: ``` FATAL EXCEPTION: main Process: com.example.app, PID: 12345 java.lang.NullPointerException: Attempt to invoke virtual method 'java.lang.String com.example.app.User.getName()' on a null object reference at com.example.app.MainActivity.onCreate(MainActivity.java:25) at android.app.Activity.performCreate(Activity.java:6975) at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1213) at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2770) at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2892) at android.app.ActivityThread.-wrap11(Unknown Source:0) at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1593) at android.os.Handler.dispatchMessage(Handler.java:105) at android.os.Looper.loop(Looper.java:164) at android.app.ActivityThread.main(ActivityThread.java:6540) at java.lang.reflect.Method.invoke(Native Method) at com.android.internal.os.Zygote$MethodAndArgsCaller.run(Zygote.java:240) at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:767) ``` 步骤三:确定异常类型和位置 崩溃日志中的第一部分(例如“java.lang.NullPointerException”)指示发生的异常类型。异常类型有助于确定崩溃的原因和所需的修复操作。接下来,您需要查看堆栈跟踪以确定崩溃的位置。在上面的示例中,堆栈跟踪指示在 MainActivity.java 的第 25 行发生了空指针异常。 步骤四:分析代码 一旦您确定了崩溃的位置,您需要分析代码以找出问题的根本原因。在上面的示例中,问题可能是在 MainActivity.java 中使用了空对象。您可以检查代码并添加必要的 Null 检查,例如: ```java User user = getUser(); if (user != null) { String name = user.getName(); // do something with name } ``` 步骤五:测试修复 一旦您修改了代码并重新编译应用程序,您需要进行测试以确保崩溃已解决。您可以重新安装应用程序并尝试重现崩溃。如果崩溃没有再次发生,则修复已成功。如果崩溃仍然存在,则需要进一步分析和修复。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值