android 内存泄漏分析与优化(二)

内存抖动、内存溢出、内存泄漏

  • 内存抖动
    在极短的时间内,分配大量的内存,然后又释放它,这种现象就会造成内存抖动。典型地,在 View 控件的 onDraw 方法里分配大量内存,又释放大量内存,这种做法极易引起内存抖动,从而导致性能下降。因为 onDraw 里的大量内存分配和释放会给系统堆空间造成压力,触发 GC 工作去释放更多可用内存,而 GC 工作起来时,又会吃掉宝贵的帧时间 (帧时间是 16ms) ,最终导致性能问题。GC工作是发生在主线程中的,因为频繁的触发GC导致掉帧就是内存抖动。
  • 内存溢出
    个Android应用程序都执行在自己的虚拟机中,每个虚拟机必定会有堆内存阈值限制(值得一提的是这个阈值一般都由厂商依据硬件配置及设备特性自己设定,没有统一标准,可以为64M,也可以为128M等;它的配置是在Android的属性系统的/system/build.prop中配置dalvik.vm.heapsize=128m即可,若存在dalvik.vm.heapstartsize则表示初始申请大小,也可以通过ActivityManager.getMemoryClass()获得这个值),也即一个应用进程同时存在的对象必须小于阈值规定的内存大小才可以正常运行,否则则会报oom(OutOfMemoryError)。产生的原因最直接的原因是一下子申请大量的内存超出阈值直接崩溃,还有原因是因为错误的程序导致内存泄漏,即使申请一小段内存也会直接崩溃。
  • 内存泄漏
    内存泄漏是本该由GC回收的内存因为某些原因得不到回收而导致的,通俗的讲是本应该被回收的对象被比它生命周期还有长的对象持有,导致不可回收,就产生了内存泄漏。
    举个例子一(单例最常见的内存泄漏):
public final class MainActivity extends Activity
 { 
    private DbManager mDbManager; 
    @Override
    protected void onCreate(Bundle savedInstanceState) 
     { 
       super.onCreate(savedInstanceState); 
       setContentView(R.layout.activity_main); 
       //DbManager是一个单例模式类,这样就持有了              
        //MainActivity引用,导致泄露 
     mDbManager = DbManager.getInstance(this); 
    }
}

分析:由于单例的静态特性使得它的生命周期比较长,又因为它持有activity,所以当activity退出时,此activity得不到GC回收从而导致了内存泄漏。

举例二:集合中

Vector v = new Vector(10);
for (int i = 1; i < 100; i++) 
{ 
    Object o = new Object(); 
     v.add(o);
     o = null; 
}

分析:
在这个例子中,我们循环申请Object对象,并将所申请的对象放入一个 Vector 中,如果我们仅仅释放引用本身,那么 Vector 仍然引用该对象,所以这个对象对 GC 来说是不可回收的。因此,如果对象加入到Vector 后,还必须从 Vector 中删除,最简单的方法就是将 Vector 对象设置为 null。

Android中常见的内存泄漏汇总以及相应的解决办法

  • 集合类
    如果仅仅有添加元素的方法,而没有相应的删除机制,导致内存被占用。如果这个集合类是全局性的变量 (比如类中的静态属性,全局性的 map 等即有静态引用或 final 一直指向它),那么没有相应的删除机制,很可能导致集合所占用的内存只增不减。比如上面的典型例子就是其中一种情况。
  • 单例造成的内存泄漏
    由于单例的静态特性使得其生命周期跟应用的生命周期一样长,所以如果使用不恰当的话,很容易造成内存泄漏。比如上面的典型例子就是其中一种情况。它的单例一般是这样
  public class AppManager 
 { 
     private static AppManager instance; 
     private Context context;
     private AppManager(Context context)
     {
       this.context = context; 
     } 
   public static AppManager getInstance(Context context) 
   { 
      if (instance == null) 
          {
            instance = new AppManager(context);
          } 
       return instance; 
   } 
}

分析:这是一个普通的单例模式,当创建这个单例的时候,由于需要传入一个Context,所以这个Context的生命周期的长短至关重要,如果此时传入的是 Application 的 Context,因为Application 的生命周期就是整个应用的生命周期,所以这将没有任何问题。如果此时传入的是 Activity 的 Context,当这个 Context 所对应的 Activity 退出时,由于该 Context 的引用被单例对象所持有,其生命周期等于整个应用程序的生命周期,所以当前 Activity 退出时它的内存并不会被回收,这就造成泄漏了。

正确的方式应该改为下面这种方式:

public class AppManager
{ 
   private static AppManager instance; 
   private Context context;
   private AppManager(Context context) 
    { 
      this.context = context.getApplicationContext(); 
    }
   public static AppManager getInstance(Context context)
    { 
        if (instance == null) 
    {
       instance = new AppManager(context);
    } 
      return instance; 
    } 
 }
  • 匿名内部类/非静态内部类和异步线程
    举个例子
public class MainActivity extends Activity 
{
...
Runnable ref1 = new MyRunable();
Runnable ref2 = new Runnable() 
{ 
@Override
 public void run()
 {
 }
};
 ...
}

分析:匿名内部类是默认持有外部的引用,因此容易造成内存泄漏。

  • Handler 使用不当造成的内存泄漏
    Handler 的使用造成的内存泄漏问题应该说是最为常见了,很多时候我们为了避免 ANR 而不在主线程进行耗时操作,在处理网络任务或者封装一些请求回调等api都借助Handler来处理,但 Handler 不是万能的,对于 Handler 的使用代码编写一不规范即有可能造成内存泄漏。另外,我们知道 Handler、Message 和 MessageQueue 都是相互关联在一起的,万一 Handler 发送的 Message 尚未被处理,则该 Message 及发送它的 Handler 对象将被线程 MessageQueue 一直持有。 由于 Handler 属于 TLS(Thread Local Storage) 变量, 生命周期和 Activity 是不一致的。因此这种实现方式一般很难保证跟 View 或者 Activity 的生命周期保持一致,故很容易导致无法正确释放。

举个栗子:


public class SampleActivity extends Activity 
{ 
    private final Handler mLeakyHandler = new Handler()
    { 
      @Override 
         public void handleMessage(Message msg) 
           { 
          // ...
           } 
   } 
     @Override 
  protected void onCreate(Bundle savedInstanceState)
     { 
       super.onCreate(savedInstanceState); // Post a message and delay its execution for 10 minutes. 
       mLeakyHandler.postDelayed(new Runnable() 
         { 
             @Override
            public void run() 
            { 
                  /* ... */
            } 
         }, 1000 * 60 * 10); // Go back to the previous Activity. 
      
     finish();
    } 
}

分析:
在该 SampleActivity 中声明了一个延迟10分钟执行的消息 Message,mLeakyHandler 将其 push 进了消息队列 MessageQueue 里。当该 Activity 被 finish() 掉时,延迟执行任务的 Message 还会继续存在于主线程中,它持有该 Activity 的 Handler 引用,所以此时 finish() 掉的 Activity 就不会被回收了从而造成内存泄漏(因 Handler 为非静态内部类,它会持有外部类的引用,在这里就是指 SampleActivity)。

修复方法:
在 Activity 中避免使用非静态内部类,比如上面我们将 Handler 声明为静态的,则其存活期跟 Activity 的生命周期就无关了。同时通过弱引用的方式引入 Activity,避免直接将 Activity 作为 context 传进去,见下面代码:


public class SampleActivity extends Activity 
{ 
/** * Instances of static inner classes do not hold an implicit * reference to their outer class. */ 
private static class MyHandler extends Handler
{ 
    private final WeakReference<SampleActivity> mActivity; 
    public MyHandler(SampleActivity activity) 
     { 
        mActivity = new WeakReference<SampleActivity>(activity); 
     }
    @Override 
     public void handleMessage(Message msg) 
    { 
        SampleActivity activity = mActivity.get();
       if (activity != null) 
          { 
             // ... 
          } 
     } 
} 
private final MyHandler mHandler = new MyHandler(this); 
/** * Instances of anonymous classes do not hold an implicit * reference to their outer class when they are "static". */
 private static final Runnable sRunnable = new Runnable() 
  { 
     @Override 
      public void run()
       {
           /* ... */
       } 
  }; 
 @Override 
protected void onCreate(Bundle savedInstanceState)
 { 
super.onCreate(savedInstanceState); // Post a message and delay its execution for 10 minutes. 
mHandler.postDelayed(sRunnable, 1000 * 60 * 10); // Go back to the previous Activity. 
finish();
 }
}

综述,即推荐使用静态内部类 + WeakReference 这种方式。每次使用前注意判空。

  • 资源未关闭造成的内存泄漏
    对于使用了BraodcastReceiver,ContentObserver,File,游标 Cursor,Stream,Bitmap等资源的使用,应该在Activity销毁时及时关闭或者注销,否则这些资源将不会被回收,造成内存泄漏。

  • 一些不良代码造成的内存压力
    比如:
    1 、Bitmap 没调用 recycle()方法,对于 Bitmap 对象在不使用时,我们应该先调用 recycle() 释放内存,然后才它设置为 null. 因为加载 Bitmap 对象的内存空间,一部分是 java 的,一部分 C 的(因为 Bitmap 分配的底层是通过 JNI 调用的 )。 而这个 recyle() 就是针对 C 部分的内存释放。
    2 、构造 Adapter 时,没有使用缓存的 convertView ,每次都在创建新的 converView。这里推荐使用 ViewHolder

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

推荐阅读更多精彩内容

  • 内存管理的目的就是让我们在开发中怎么有效的避免我们的应用出现内存泄漏的问题。内存泄漏大家都不陌生了,简单粗俗的讲,...
    宇宙只有巴掌大阅读 2,360评论 0 12
  • Android 内存泄漏总结 内存管理的目的就是让我们在开发中怎么有效的避免我们的应用出现内存泄漏的问题。内存泄漏...
    _痞子阅读 1,622评论 0 8
  • 被文同时发布在CSDN上,欢迎查看。 APP内存的使用,是评价一款应用性能高低的一个重要指标。虽然现在智能手机的内...
    大圣代阅读 4,803评论 2 54
  • Android 内存泄漏总结 内存管理的目的就是让我们在开发中怎么有效的避免我们的应用出现内存泄漏的问题。内存泄漏...
    apkcore阅读 1,216评论 2 7
  • 内存管理的目的就是让我们在开发中怎么有效的避免我们的应用出现内存泄漏的问题。内存泄漏大家都不陌生了,简单粗俗的讲,...
    DreamFish阅读 790评论 0 5