Android的单例模式在我们项目开发中经常会用到,不过使用的不恰当的话也会造成内存泄漏。因为单例的静态特性使得单例的生命周期和应用的生命周期一样长, 这就说明了如果一个对象已经不需要使用了,而单例对象还持有该对象的引用,那么这个对象将不能被正常回收,这就导致了内存泄漏。
**粗暴点说**,长生命周期的对象持有短生命周期的对象,短周期对象就无法及时释放。
我们举个例子:
新建一个工程。
添加一个单例类MyDemo,代码如下
/**
* Created by Zach on 2017/3/31.
*/
public class MyDemo {
private static MyDemo ourInstance;
private Context context;
//看到了这里注意了context参数
public static MyDemo getInstance(Context context) {
if (ourInstance==null){
synchronized (MyDemo.class){
if (ourInstance==null){
ourInstance=new MyDemo(context);
}
}
}
return ourInstance;
}
private MyDemo(Context context) {
this.context=context;
}
}
- 在MainActivity中运行 调用
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
MyDemo myDemo=MyDemo.getInstance(this);
}
}
- 返回键退出到主页 ,几秒后就leak啦
————————————华丽分割线————————————
为啥呢,究竟问啥呢?
MyDemo myDeom=MyDemo.getInstance(this);
这句传入的是Activity的Context,我们都知道,Activty是间接继承于Context的,当这Activity退出时,Activity应该被回收, 但是单例中又持有它的引用,导致Activity回收失败,造成内存泄漏。
因为返回键退出,act被回收,单例模式的MyDemo还在运行,也就是文章开头说的 长生命周期的对象持有短生命周期的对象,短周期对象就无法及时释放,造成leak。
也就是大家常说的,context持久引用(act被收回了,尼玛没被回收的单例还引用,当然泄露了)。
类似的还有,非静态内部类创建静态实例造成的内存泄漏(Context持久引用)Handler造成的内存泄漏(Context持久引用、Message对象引用)…..