内存泄露背景
什么是内存泄露
内存空间使用完毕后无法被释放的现象,对于还保持着引用, 该内存不能再被分配使用,逻辑上却已经不会再用到的对象,垃圾回收器不会回收它们。
所以逻辑不再使用的对象,需要释放强引用,以便GC进行回收。
JVM 工作原理
JVM 垃圾回收原理,点击进入
内存泄露案例
- 常见Handler 写法
Handler mHandler = new Handler(){
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
switch (msg.what) {
case 0:
// do something
sendEmptyMessageDelayed(100, 2000);
break;
case 1:
// do something
break;
default:break;
}
}
};
mHandler.sendEmptyMessage(1);
This Handler class should be static or leaks might occur (anonymous android.os.Handler)
- Handler 内存泄漏说明
mHandler 会作为成员变量保存在发送的消息msg 中,即 msg 持有mHandler 的引用,而 mHandler 是 Activity 的非静态内部类实例,即mHandler 持有 Activity 的引用,那么我们就可以理解为 msg 间接持有 Activity 的引用。msg 被发送后先放到消息队列 MessageQueue 中,然后等待 Looper 的轮询处理(MessageQueue 和 Looper 都是与线程相关联的, MessageQueue 是 Looper 引用的成员变量,而 Looper 是保存在ThreadLocal 中的)。那么当 Activity 退出后,msg 可能仍然存在于消息对列MessageQueue 中未处理或者正在处理,那么这样就会导致 Activity 无法被回收,以致发生Activity 的内存泄露。
通常在Android 开发中如果要使用内部类,但又要规避内存泄露,一般都会采用 静态内部类 + 弱引用 的方式。
内存泄露优化
- 自定义一个静态内部类 Handler
用静态内部类的方式自定义一个 Handler ,使用 WeakReference 实现弱引用。
传入的 SecondActivity 是使用这个 Handler 的 Activity ,按照实际情况传入即可; 传入Fragment 也是可以的。
private static class SecondHandler extends Handler{
private WeakReference<SecondActivity> mWeakReference;
public SecondHandler(SecondActivity secondActivity) {
mWeakReference = new WeakReference<>(secondActivity);
}
@Override
public void handleMessage(Message msg) {
SecondActivity host = mWeakReference.get(); // 判断所在的 Activity 的引用是否被回收了
if (host != null) {
switch (msg.what){
case 99:
// do something
sendEmptyMessageDelayed(100, 2000);
break;
case 100:
// do something
break;
default:break;
}
}
}
}
mHandler 通过弱引用的方式持有Activity,当GC 执行垃圾回收时,遇到Activity 就会回收并释放所占据的内存单元。这样就不会发生内存泄露了。文章来源:https://www.toymoban.com/news/detail-718098.html
- 弱引用 Handler 初始化
参考构造函数,初始化也很简单,使用的话还是 Handler 的用法。
SecondHandler mHandler = new SecondHandler(SecondActivity.this);
mHandler.sendEmptyMessage(1);
- 退出时移除回调和消息
当 Activity 退出后,将 mHandler 的回调和消息移除。在 onStop 或者 onDestroy 中执行即可。
@Override
protected void onStop() {
super.onStop();
mHandler.removeCallbacksAndMessages(null);
}
传入 null 是移除所有回调和消息,查看 Handler 源码可知,文章来源地址https://www.toymoban.com/news/detail-718098.html
/**
* Remove any pending posts of callbacks and sent messages whose
* <var>obj</var> is <var>token</var>. If <var>token</var> is null,
* all callbacks and messages will be removed.
* */
public final void removeCallbacksAndMessages(Object token) {
mQueue.removeCallbacksAndMessages(this, token);
}
到了这里,关于【Android内存优化】内存泄露优化之强引用变弱引用完全详解的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!