为什么定义一个static ImageView会导致内存泄露?

drunkpiano
  • 176

提示说:不要把「Do not place android context classes in static fields, this is a memory leak.」

图片描述

我知道activity context 不应该被放到static fields里面因为它在app生命周期内一直存在,而且如果它通过构造函数传递给其他class的话,其他class也会一直不被回收。

但为什么ImageView不能是static(我发现SwipeRefreshLayout也不行)?是因为ImageView包含contextButtonTextView之类的view就没有这种问题。

回复
阅读 6.1k
7 个回答

在Activity中填充布局,可以通过如下方式:

1.view view = View.inflate(context, R.layout.test,null );
2.View view = LayoutInflater.from(context).inflate(R.layout.test,null);

可以发现,无论哪种方式,填充后的View都是持有context的引用的,也就是Activity的引用.
如果此时填充的view是static修饰,那么静态的对象将持续持有Activity的引用,导致Activity无法销毁,Activity中所有的控件也都将无法彻底销毁和回收.最终造成内存泄露.

PS:为什么要使用静态View的方式?有什么特殊的业务需求?

static变量在内存中是单独存在于内存块中的,控件(你的问题中指的是ImageView)是持有Activity的引用的,这种情况下,Activity是没法被彻底销毁的,因为在内存中一直有一个引用(指针理解),导致Activity也无法被回收,自然就会内存泄漏了!
建议,在Android中不要使用static修饰控件!

这么说吧,这是java基础.静态变量在类文件编译成字节码后,载入方法区的时候,就会初始化静态变量,而且静态变量会一直存在,而没有static修饰的activity,会在new的时候,在堆中申请空间,当activity没用的时候,gc会按照gc-root遍历此activity有没有被引用,由于activity的context被持有,ImageView view = LayoutInflater.from(context).inflate(R.layout.test,null);,所以,gc不能回收,造成内存泄漏,,当直到虚拟机结束进程(也就是app停止),

View需要持有Context (否则无法实现 setXXX(int resid)类的功能)

我也不知道Button/TextView为什么不警告

static变量无法被自动回收啊
你将某个view设为静态量,而view又持有activity的引用,这下连activity也泄露了

IPC说: static对于Android四大组件来说, 简直就是大杀器

因为static的生命周期很长比当前activity长

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
宣传栏