在拜读包建强前辈著作的《App研发录》过程中,我深有感触。因为在书中提到的一些crash(崩溃),我也经常遇到。特此,我结合自己遇到的困扰与书中描述的问题,提出个人解决方案或者如何避免此类问题发生。

1、NullPointException(空指针异常)

原因是传入空指针,导致程序崩溃。下面介绍三种相关情况:

(1)在对网络请求返回的结果进行处理时,如果直接解析返回空json,那么app就可能崩溃。所以需要在处理接口返回数据加上非空判断或者try-catch语句。

(2)在数据库查询得到cursor对象时,直接调用cursor.hasNext()方法,如果cursor为空,那么app可能发生异常。所以应该在调用该方法前加cursor的非空判断。另外补充下:使用cursor完毕,不要忘记调用cursor.close()方法来关闭它。

(3)在使用new File(filePath)时,如果filePath不存在就可能报错。所以应该先判断filePath是否存在。

2、StringIndexOutOfBoundsException/ArrayIndexOutOfBoundsException(字符串或数组下标越界)

对于StringIndexOutOfBoundsException,往往是截取字符串超出原字符串长度,那么应该预判断原字符串长度,保证start与end不超出原长度。对于ArrayIndexOutOfBoundsException,对应有两种常见情况:遍历数组时,没对数组进行非空判断以及长度是否大于0;使用数组时,没预判断数组长度,使得数组下标越界。

3、Attempt to invoke virtual method on a null object reference(在一个空对象引用,尝试调用不存在方法)

这种crash的产生,是因为在使用一个对象的某个方法时,该对象为空,也就是没有实例化。那么,我们在调用对象的方法前,需要先判断对象是否为空,确保对象被实例化了。

4、ClassCastException/NumberFormatException(数据类型转换异常/数字转换错误)

(1)在数据类型转换过程中,如果转换不成功,一般抛出ClassCastException。例如Object x = new Integer(0);String str = (String)x;这样会抛出该异常了。解决方案是,使用安全类型转换函数,判断数据源是否为空,并且如果转换出错,返回默认值。

(2)当字符串转换成数字失败时,会抛出NumberFormatException。例如:String str = "12xy345"; int res = Integer.parseInt(str);

因为字符串数据源里面包含字符,所以无法转换成数字,抛出异常。那么,应该加try-catch处理,并且转换失败时返回默认值。

5、ConcurrentModificationException(并发修改异常)

多个线程修改同一个集合的元素,引起的异常。解决是在多线程操作共享变量加同步锁。

6、Comparison method violates its general contract(比较器违法基本原则)

这个错误是因为使用Comparison的compare方法不当引起的。例如:

Comparator<int> comp = new Comparator<int> {
   public int compare(int a, int b){
       return a > b ? 1 : -1; 
   }
};


这里是因为少考虑a等于b的情况。正确用法是:


Comparator<int> comp = new Comparator<int> {
   public int compare(int a, int b){
       if(a > b){
          return 1;
       }else if(a < b){
          return -1;
       }else{
          return 0;
       }
   }
};



7、NoClassDefFoundError(没有找到类定义的错误)

此种情况往往dex分包导致的。我碰到的一种情况:Android5.0及以上的手机没有报错,而5.0以下的手机抛出该异常。原因是方法数超过65535个了,该方法无法在主dex找到,而Android5.0以上默认支持多dex分包,5.0以下不支持。所以应该在主模块gradle的defaultConfig里面加上:multiDexEnabled true。并且你的application继承MultiDexApplication。

8、ActivityNotFoundException(找不到Activity异常)

出现错误原因是Intent里传递的URL不是以“http”开头的。还有一种情况是,要启动第三方APP的activity,而手机并没有安装对应的APP,所以无法找到对应Activity。

9、Fragment not attached to Activity(Fragment没有关联到Activity异常)

发生这个异常,是因为Fragment还没有Attach到Activity,就调用诸如getResource()方法。解决方案是:在获取资源之前先用isAdded()方法判断,如果为true才允许调用获取资源方法。

10、Parcelable encountered IOException writing serializable object(parcelable 写序列化对象时遇到异常)

原因是要序列化对象里,有些不支持序列化:实体、JSONOjbject、JSONArray等。

11、The content of the adapter has changed but ListView did not receive a notification(adapter数据发生改变但列表没更新)

报错原因是adapter内容变化了,但是对应的ListView没有更新。请保证adapter的数据是在主线程进行修改的。解决方法有如下三种:

(1)调用Activity的runOnUiThread()方法;

(2)调用handler,通知主线程修改adapter;

(3)调用AsyncTask异步任务。

12、IndexOutOfBoundsException:Invalid index 30,size is 0.(数组下标越界异常)

此问题往往是ListView或者类似列表处于滚动时点击刷新造成的。因为listView滚动时,表示它已经获取了adapter的getCount()为30。但是回调getView()后,数据被clear清空了,所以抛出该异常。对应解决方法是,当listView处于滚动时,设置刷新按钮不可点击。

13、IllegalArgumentException:View not attached to window manager(非法参数异常,视图没有关联到窗体管理器)

发生此类异常的一种情景:执行一个耗时任务,任务开始时显示对话框,任务完成后关闭对话框。如果在此期间Activity由于某种原因被销毁重新创建,那么当Dialog调用dismiss方法时,WindowManager发现Dialog所属的Activity已经不存在,所以抛出此异常。解决方法是:调用dismiss方法关闭对话框前,先判断Dialog所属的Activity是否存在并且Dialog是否处于open状态。避免此问题的更好方案是,使用Activity相应的操作对话框回调:onCreateDialog、showDialog、dismissDialog、removeDialog,确保与Activity的生命周期一致。

14、CalledFromWrongThreadException(调用的线程不对引起异常)

此类问题往往是在子线程修改UI引起的,而android系统规定只有主线程才能执行更新界面操作。比如在子线程调用tv.setText();

15、StackOverFlow(栈溢出)

发生该异常的原因是由于无限递归入栈,造成栈溢出。有种情况是:在App退出时,有多个线程在运行,调用finish方法,无法及时全部关闭。那么,需要调用System.exit(0)方法来退出。

16、UnsatisfiedLinkError:dalvik.system.PathClassLoader(不满足链接错误)

出现该异常是因为无法加载到对应的so库。我碰到的一种情况是:工程中jniLibs目录下只有armeabi架构的so文件。使用android5.0以上的某些机型就会抛出该异常,因为这些机型是armeabi-v7a架构。后来在jniLibs目录下添加armeabi-v7a架构对应so文件,才恢复正常。

好了,直到这里已经介绍16种常见的crash,下篇再继续补充。希望对遇到类似crash的开发者有所帮助,或者是刚进入android

开发领域的同行看后,能够尽量避免这些crash的发生。