搜索

首页  >  问答  >  正文

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

提示说:不要把「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就没有这种问题。

PHPzPHPz2766 天前731

全部回复(7)我来回复

  • 迷茫

    迷茫2017-04-18 09:08:00

    在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的方式?有什么特殊的业务需求?

    回复
    0
  • PHP中文网

    PHP中文网2017-04-18 09:08:00

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

    回复
    0
  • 阿神

    阿神2017-04-18 09:08:00

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

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

    回复
    0
  • 怪我咯

    怪我咯2017-04-18 09:08:00

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

    回复
    0
  • 迷茫

    迷茫2017-04-18 09:08:00

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

    回复
    0
  • 阿神

    阿神2017-04-18 09:08:00

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

    回复
    0
  • 怪我咯

    怪我咯2017-04-18 09:08:00

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

    回复
    0
  • 取消回复