android 内存优化篇之内存泄漏原因与内存泄漏优化

1.内存泄漏原因:

Java内存泄漏指的是进程中某些垃圾对象已经没有使用价值了,但是它们却可以直接或间接地引用到gc roots导致无法被GC回收。无用的对象占据着内存空间,使得实际可使用内存变小,形象地说法就是内存泄漏了。

2.内存泄漏优化:

内存泄漏的优化分为两个方面,一方面实在开发过程中避免写出泄漏的代码,另一方面则是通过一些分析工具MAT,LeakCanary来找出潜在的内存泄漏继而解决。

3.常见内存泄漏:

1:静态变量导致的内存泄漏,下面的情况就是activity无法正常销毁,因为静态变量Context引用了它。

   public class MainActivity extends Activity {
   private static Context mContext;
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mContext = this;
    }
}

上面的代码我们也可以改造一下,mView是一个静态变量,他的内部持有当前的activity,所以activity仍然无法释放,估计这样好理解点。

public class MainActivity extends Activity {

private static View mView;

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    mView = new View(this);
}
}

2:非静态内部类的静态实例导致内存泄漏,下面的情况可以看得出静态实例,一直持有当前的activity的引用,我们称为act1吧。当act1 onDestory()的时候并没有被回收,当我们再次创建的时候会出现act2,进程中就会出现两个act,因此对于launchMode不是singleInstance的activity,因避免这种情况

   public class MainActivity extends Activity {

    private static ExampleDemo demo;
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        demo = new ExampleDemo();
    }
}

class ExampleDemo{
  void doSomeThing(){
            ...
  }
} 

3:单例模式,单例模式应该是设计模式被开发者所喜爱的设计模式,使用简单粗暴。但是单例模式所带来的内存泄漏是我们容易忽视的,如下所示提供一个单例模式

 public class OrmHelper {
    private Context context;

    private static OrmHelper ormHelper;

    public OrmHelper(Context context) {
        this.context = context;
    }

    public static OrmHelper getInstance(Context context) {
        synchronized (OrmHelper.class) {
            if (ormHelper == null) {
                ormHelper = new OrmHelper(context);
            }
        }
        return ormHelper;
    }
}

泄漏的原因是OrmHelper会一直持有context对象,导致当前的activity无法被回收,而单例模式的特点是其生命周期和application保存一直,因此activity对象无法回收。下面是单例模式的优化。使用弱引用。

public class OrmHelper {
    private WeakReference<Context> context;

    private static OrmHelper ormHelper;

    public OrmHelper(Context context) {
        this.context = new WeakReference<Context>(context);
    }

    public static OrmHelper getInstance(Context context) {
        synchronized (OrmHelper.class) {
            if (ormHelper == null) {
                ormHelper = new OrmHelper(context);
            }
        }
        return ormHelper;
    }
}

也可以有另一种方式,就是如果这个工具类在app中进程经常被使用可以传入ApplicationContext。

4:属性动画导致内存泄漏,在android3.0开始Google提供了属性动画,属性动画中又有一种无线循环的动画,如果在activity中播放此类动画,没有在ondestory中进行停止动画,那么动画会一直播放,尽管无法再页面上看到动画效果,但activity的view 会被动画持有,而view又持有activity。最终导致activity无法被释放。解决方法就是在onDestory中调用animator.cancle();下面是一个例子

public class MainActivity extends Activity {

    private Button mButton;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mButton = ((Button) findViewById(R.id.sub));

        ObjectAnimator animator = ObjectAnimator.ofFloat(mButton,"rotation",0,360).setDuration(2000);
        animator.setRepeatCount(ValueAnimator.INFINITE);
        animator.start();
//        animator.cancel();
    }
}

5:注册某个对象后没有反注册,假设我们要监听电话状态,需要创建个PhoneStateListener,同时要将它注册到telephoyManager中。

public class MainActivity extends Activity {
    private TelephonyManager tm;
    private PhoneStateListener mPhoneStateListener = new PhoneStateListener() {
        @Override
        public void onCallStateChanged(int state, String incomingNumber) {
            switch (state) {
                case TelephonyManager.CALL_STATE_IDLE:
                    break;

                case TelephonyManager.CALL_STATE_RINGING: {
                    break;
                }
                case TelephonyManager.CALL_STATE_OFFHOOK: {
                    break;
                }
            }
            super.onCallStateChanged(state, incomingNumber);
        }
    };

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        tm = (TelephonyManager)getSystemService(Service.TELEPHONY_SERVICE);
        tm.listen(mPhoneStateListener,PhoneStateListener.LISTEN_CALL_STATE);
    }
}

这种情况就回出现很严重的内存泄漏,没有反注册。我们在onCreate中进行了注册,则需要在OnDestory()进行反注册,同时把activity的引用缓存弱引用,尽量来避免内存泄漏。

public class MainActivity extends Activity {
    private TelephonyManager tm;
    private PhoneStateListener mPhoneStateListener = new PhoneStateListener() {
        @Override
        public void onCallStateChanged(int state, String incomingNumber) {
            switch (state) {
                case TelephonyManager.CALL_STATE_IDLE:
                    break;

                case TelephonyManager.CALL_STATE_RINGING: {
                    break;
                }
                case TelephonyManager.CALL_STATE_OFFHOOK: {
                    break;
                }
            }
            super.onCallStateChanged(state, incomingNumber);
        }
    };
    private WeakReference<Context> weakReference;
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        weakReference = new WeakReference<Context>(this);
        new Thread(new Runnable() {
            @Override
            public void run() {
                tm = (TelephonyManager)weakReference.get().getSystemService(Service.TELEPHONY_SERVICE);
                tm.listen(mPhoneStateListener,PhoneStateListener.LISTEN_CALL_STATE);
            }
        }).start();

    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        tm.listen(mPhoneStateListener, PhoneStateListener.LISTEN_NONE);
    }
}

在这里我们做了两方面优化,1.引用缓存弱引用,2.创建telephonyManager的时候在线程中创建(通过dump 最终发现是 PhoneStateListener 内部对自己有一个强引用的handler,如果是在主线程中引用PhoneStateListener,那么他将释放不掉,引发内存泄露)。

6:回调接口,android 4.0已经使用弱引用来解决了这个问题

 private WeakReference<IMusicCallBack> callBackWeakReference;
    public void setCallBack(IMusicCallBack callBack){
        callBackWeakReference = new WeakReference<IMusicCallBack>(callBack);
    }

7:集合对象没有清理造成内存泄漏,我们通常把一些对象的引用加入到了集合中,当我们不需要该对象时,如果没有把它的引用从集合中清理掉,这样这个集合就会越来越大。如果这个集合是static的话,那这种内存泄漏情况就更严重了。

8:资源对象没有手动关闭或处理,资源性对象比如(Cursor,File文件等)往往都用了一些缓冲,我们在不使用的时候,应该及时关闭它们,以便它们的缓冲及时回收内存。它们的缓冲不仅存在于 java虚拟机内,还存在于java虚拟机外。如果我们仅仅是把它的引用设置为null,而不关闭它们,往往会造成内存泄漏。因为有些资源性对象,比如 SQLiteCursor(在析构函数finalize(),如果我们没有关闭它,它自己会调close()关闭),如果我们没有关闭它,系统在回收它时也会关闭它,但是这样的效率太低了。因此对于资源性对象在不使用的时候,应该调用它的close()函数,将其关闭掉,然后才置为null.在我们的程序退出时一定要确保我们的资源性对象已经关闭。

9:Bitmap使用不当1.及时的销毁给bitmap分配的内存,recycle。2.有些时候我们需要显示的图片区域很小,没有必要把原图全部显示出来,而只需要显示缩小过得图片,降低采样率就行了。这样大大的减少了内存的使用。

4.总结

在开发中内存泄漏问题,还会有很多种情况,开发中尽量避免写出有内存泄漏的代码,但并不是每个内存泄漏都需要解决,我们只是尽量的让内存泄漏减少,来优化app内存。希望对广大读者开发有好的帮助。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,393评论 5 467
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,790评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,391评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,703评论 1 270
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,613评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,003评论 1 275
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,507评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,158评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,300评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,256评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,274评论 1 328
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,984评论 3 316
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,569评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,662评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,899评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,268评论 2 345
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,840评论 2 339

推荐阅读更多精彩内容