本文主要总结Android开发中发生的内存泄漏场景,发生的原因,以及解决方案。
目录
- Java内部类隐式持有外部类
- Java匿名内部类隐式持有外部类
- Android的Window类持有了activity
- [LeakCanary提示
policy.HwPhoneWindow$1.this$0
的泄漏](#LeakCanary提示policy.HwPhoneWindow$1.this$0
的泄漏) - [Android InputMethodManager](#Android InputMethodManage)
java的gc检查内存回收时,使用有向图机制,及检查一个或一组对象的可达性。如果是不可达状态,则将该对象从内存中回收,所谓不可达状态,也就是一个对象不被任何引用所指向或者叫持有。
另外,java的循环引用不同于oc或者c++,当一组对象中只包含互相的引用,而没有来自它们外部的引用(例如有两个对象A和B互相持有引用,但没有任何外部对象持有指向A或B的引用),这仍然属于不可到达,同样会被GC回收。
注意:
在Android开发过程中,activity会天然的被window对象所应用,所以在activity中与其他类循环应用时,尽量使用弱引用模式,
Java内部类隐式持有外部类
常见于
ViewHolder
或者new Handler
,new Listener
等情况
案例如下:
public class MainActivity extends AppCompatActivity {
private TestResource mResource = null;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
if(mManager == null){
mManager = new TestResource();
}
//...
}
class TestResource {
//...
}
}
分析:
如果TestResource
我们执行了耗时操作,在销毁Activity时,由于testResource隐藏性的指向了Activity的实例,所以在testResource会执行结束时,不会销毁Activity,发生了内存泄漏。
解决方案:
内部类增加static前缀,这样就不会隐式指向外部类了。增加static前缀也相当了新建一个java文件
static class TestResource {
//...
}
}
Java匿名内部类隐式持有外部类
该案例同上书问题一样,我们为了方便会在大量使用匿名内部类,而不是重新定义内部类,比如在在Thread或者事件监听场景中。
新建一个线程
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
new Thread(new Runnable(){
@Override
public void run() {
}
}).start();
}
分析:
此时如果发生在Activity类,比我们可以看到在run()函数可以直接调用activity的成员变量,说明这个Runnable匿名内部类已经持有了Activity的实例,当Runnable执行耗时操作时,Activity就销毁不掉了。
解决方案:
匿名内部类提出来,作为成员变量并增加static前缀,就解除了与外外部类的关系
static Runnable runnable = new Runnable() {
@Override
public void run() {
}
}
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
new Thread(runnable).start();
}
LeakCanary提示policy.HwPhoneWindow$1.this$0
的泄漏
最新在华为手机GEM=703L android6.0发现的问题,在
AsyncTask
执行ProgressDialog
的显示或隐藏,然后退出activity会发生泄漏。泄漏提示
GC ROOT com.android.internal.policy.HwPhoneWindow$1.this$0
分析过程太长了,转移到新的文章
LeakCanary提示policy.HwPhoneWindow$1.this$0
的泄漏分析
Android InputMethodManager
这中情况导致的内存目前还在研究中,先贴出大家的讨论