Android疑难bug统计

761 阅读3分钟

1.java.lang.UnsatisfiedLinkError

多半是so加载姿势不对。so没加载进来或者so初始化失败。把三方库的初始化放回application试试
建议:使用的minSdkVersion对应的NDK版本来编译so。因为NDK是向上兼容的
提示:21以上用Build.SUPPORTED_ABIS[0],21一下用Build.CPU_ABI获取设备CPU ABI。其中Build.CPU_ABI能表示系统为这个应用选择的ABI。

2.java.lang.ArrayIndexOutOfBoundsException

数组越界,普通的数组越界还是比较容易发现的。下面场景的数组越界不容易被发现。
1.下拉刷新数据还没有返回时,迅速向上滑。
主要还是写法错误,在数据还没有返回时就删除了数据源,导致向上滑动时数据源已经没有了。

2.下拉刷新数据还没有返回时点击item。
错误的写法和上面是一样的

3.需要使用网络返回的数据position,而网络请求数据没有返回
建议:刷新数据清空放到网络数据请求的回调中。page为1清空数据;page不为1不清空。

3.java.lang.NullPointerException

空指针异常,普通的空指针异常也是比较容易发现的。下面的场景比较难发现。
网络请求没有返回导致数据实体为空,而在点击或者其他操作中又使用了该实体。
建议:在所有使用网络返回的数据实体的地方加上判空处理。

4.Unable to add window -- token android.os.BinderProxy@1fb9679 is not valid; is your activity running?

dialog所依附的view已经不存在
建议:在dialog.show之前判断activity.isFinishing()。
5.android.view.WindowLeaked: Activity com.mistong.ewt360.ui.activity.MainActivity has leaked window
当面页面关闭了,但是依附在当前页面的dialog却没有被关闭,造成内存泄漏。
建议:在onDestory加入if (mDialog!=null && mDialog.isShowing()) { mDialog.dismiss(); }
提示:多用DailogFragment,便于生命周期的管理。

6.Unable to add window -- token android.app.LocalActivityManager$LocalActivityRecord @xxx is not valid; is your activity running?

参数context 指定成了this,即指向当前子Activity的context。
但子Activity是动态创建的,不能保证一直存在。
建议:将context替换为getParent。

7.Fragment already added: KnowledgeTreeDialogFragment

Fragment重复添加,小心DialogFragment也属于fragment,在show时会调用add方法
建议:Fragment加FragmentManager.findFragmentByTag和isAdded判断DialogFragment重写show方法。
 override fun show(manager: FragmentManager?, tag: String?) {
        try {
            if (manager == null || manager.isDestroyed) {
                return
            }
            manager.beginTransaction().remove(this).commit()
            super.show(manager, tag)
        } catch (e: Exception) {
            //防止onSaveInstanceState之后执行add fragment
            e.printStackTrace()
        }
    }

8.Can not perform this action after onSaveInstanceState

Fragment在onSaveInstanceState做操作就会报这个错误,
把commit换成commitAllowingStateLoss可解决这个问题,会忽略mStateSaved。
建议:用commitAllowingStateLoss需要检查功能正确性,commit操作尽量放在onCreate方法中

9.handler建议用静态类和弱引用的方式使用,防止内存泄漏

10.java.lang.IllegalStateException intent.getStringExtra("orderNo") must not be null

从表意上看是Activity跳转的时候orderNo没有传,实际开发中这么傻逼的行为应该比较少。
我这边的场景是 小米 MI 5S 微信支付回来会重新去intent中取orderNo,这时候的intent是微信支付发的,
里面是没有orderNo的。所以导致这个报错。
建议:intent取字符串加是否有这个字段判断(因为字符串字段是没有默认值的), if(intent.hasExtra("orderNo")) intent.getStringExtra("orderNo") else ""

11.TransactionTooLargeException

Intent传参数据量偏大,bundle缓冲区最大1M
建议:用Eventbus等其他方式代替

12.不要用ScrollView嵌套RecyclerView、ListView。这样会把item一次性加载到内存中。推荐使用NestedScrollView。

13.多线程操作数据库需要使用事务。

14.页面关闭关闭动画。

15.回调看情况需要判断一下对象控件是否还存在。

16.网络相关对象不要用lateinit