Android 内存溢出 内存泄漏(二)

首先参考 Android 内存溢出 内存泄漏(一)

1. hashset的OOM

2. Cursor对象 流对象Stream未正确关闭
解决方案:finally中关闭

    Cursor cursor = null;
    try {
        cursor = getContentResolver().query(...);
        if (cursor != null && cursor.moveToNext()) {
        ... ...
        }
    } catch (Exception e) {
        ... ...
    } finally {
        if (cursor != null) {
                cursor.close();
        }
    }

3. bitmap不用,低版本中没有调用recycle发生OOM
高版本中就不需要了,因为像素和对象都存储在程序内存中。由GC回收。但是在低版本中bitmap像素放在本地内存中,必须手动调用recycle才能回收

3. createBitmap创建bitmap时,没有足够的内存分配的时候发生OOM
一般的处理方案是createBitmap的时候,捕获OutOfMemoryError,而不是Exception这样是无效的

4. 构造listview的Adapter时,没有使用缓存的convertView

5. 没有正确注册/注销监听器对象的造成OOM
- registerXxx和unregisterXxx方法的调用通常也和Cursor的打开/关闭类似,在Activity的生命周期中成对的出现即可:
在 onCreate() 中 register,在 onDestroy() 中 unregitster;
在 onStart() 中 register,在 onStop() 中 unregitster;
在 onResume() 中 register,在 onPause() 中 unregitster;

  • 忘记unregister
    以前看到过一段代码,在Activity中定义了一个PhoneStateListener的对象,将其注册到TelephonyManager中:
    TelephonyManager.listen(l,PhoneStateListener.LISTEN_SERVICE_STATE);
    但是在Activity退出的时候注销掉这个监听,即没有调用以下方法:
    TelephonyManager.listen(l,PhoneStateListener.LISTEN_NONE);
    因为PhoneStateListener的成员变量callback,被注册到了TelephonyRegistry中,TelephonyRegistry是后台的一个服务会一直运行着。所以如果不注销,则callback对象无法被释放,PhoneStateListener对象也就无法被释放,最终导致Activity对象无法被释放。

注册/反注册,监听/去监听成对出现是个好习惯,生命周期匹配中去出现即可

        //获取view的大小
        volleyImageview.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
            @Override
            public void onGlobalLayout() {
                QDLog.d("button width:" + volleyImageview.getMeasuredWidth() + " height:" + volleyImageview.getMeasuredHeight());
                volleyImageview.getViewTreeObserver().removeOnGlobalLayoutListener(this);
            }
        });

    @Override
    protected void onStart() {
        super.onStart();
        registerReceiver(BroadcastReceiver receiver, IntentFilter filter);
    }

    @Override
    protected void onStop() {
        super.onStop();
        unregisterReceiver(BroadcastReceiver receiver);
    }

查看源码发现,虽然上面两个不去注册/去监听,并不会导致activity不能被GC回收,但是注册/反注册,监听/去监听成对出现还是个编程的好习惯

6. 无线循环的动画,造成内存泄漏
内存泄漏,view被无限循环动画持有,view持有context,所以acitivity不能回收
animtor.setRepeatCount(Animation.INFINITE);
解决方案:animtor.cancel();
离开的时候,记得手动取消或停止

7. webview导致的内存泄漏
使用 WebView 的页面(Activity),在生命周期结束页面退出(onDestory)的时候,需要主动调用 WebView.onPause()以及 WebView.destory()以便让系统释放 WebView 相关资源。
建议另外启动一个进程专门运行 WebView。不要9998,不要9999,我们要100%!WebView 用完之后就把进程杀死,即使泄漏了也无碍。
参考这里 Android WebView Memory Leak WebView内存泄漏

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值