`

Android下常见的内存泄漏

 
阅读更多

 1、非静态内部类的静态实例

1.  public class MainActivity extends Activity  

 

2.  {  

 

3.      static Demo sInstance = null;  

 

4.      @Override  

 

5.      public void onCreate(BundlesavedInstanceState)  

 

6.      {  

 

7.          super.onCreate(savedInstanceState);  

 

8.          setContentView(R.layout.activity_main);  

 

9.          if (sInstance == null)  

 

10.        {  

 

11.           sInstance= new Demo();  

 

12.        }  

 

13.    }  

 

14.    class Demo  

 

15.    {  

 

16.        voiddoSomething()  

 

17.        {  

 

18.           System.out.print("dosth.");  

 

19.        }  

 

20.    }  

 

21.}  

 

上面的代码中的sInstance实例类型为静态实例,在第一个MainActivityact1实例创建时,sInstance会获得并一直持有act1的引用。当MainAcitivity销毁后重建,因为sInstance持有act1的引用,所以act1是无法被GC回收的,进程中会存在2个MainActivity实例(act1和重建后的MainActivity实例),这个act1对象就是一个无用的但一直占用内存的对象,即无法回收的垃圾对象。所以,对于lauchMode不是singleInstance的Activity,应该避免在activity里面实例化其非静态内部类的静态实例。

 

2、activity使用静态成员

 

1.  private static Drawable sBackground;    

 

2.  @Override    

 

3.  protected void onCreate(Bundle state) {    

 

4.      super.onCreate(state);    

 

6.      TextView label = new TextView(this);    

 

7.      label.setText("Leaks are bad");    

 

9.      if (sBackground == null) {    

 

10.        sBackground = getDrawable(R.drawable.large_bitmap);    

 

11.    }    

 

12.    label.setBackgroundDrawable(sBackground);    

 

14.    setContentView(label);    

 

15.}   

 

由于用静态成员sBackground 缓存了drawable对象,所以activity加载速度会加快,但是这样做是错误的。因为在android 2.3系统上,它会导致activity销毁后无法被系统回收。label.setBackgroundDrawable函数调用会将label赋值给sBackground的成员变量mCallback。上面代码意味着:sBackground(GC Root)会持有TextView对象,而TextView持有Activit对象。所以导致Activity对象无法被系统回收。

 

以上2个例子的内存泄漏都是因为Activity的引用的生命周期超越了activity对象的生命周期。也就是常说的Context泄漏,因为activity就是context。 

 

想要避免context相关的内存泄漏,需要注意以下几点:

 

·不要对activity的context长期引用(一个activity的引用的生存周期应该和activity的生命周期相同)

 

·如果可以的话,尽量使用关于application的context来替代和activity相关的context

 

·如果一个acitivity的非静态内部类的生命周期不受控制,那么避免使用它;正确的方法是使用一个静态的内部类,并且对它的外部类有一WeakReference,就像在ViewRootImpl中内部类W所做的那样。

 

3、使用handler时的内存问题

 

我们知道,Handler通过发送Message与主线程交互,Message发出之后是存储在MessageQueue中的,有些Message也不是马上就被处理的。在Message中存在一个 target,是Handler的一个引用,如果Message在Queue中存在的时间越长,就会导致Handler无法被回收。如果Handler是非静态的,则会导致Activity或者Service不会被回收。所以正确处理Handler等之类的内部类,应该将自己的Handler定义为静态内部类。

 

HandlerThread的使用也需要注意:

 

  当我们在activity里面创建了一个HandlerThread,代码如下:

 

1.  public class MainActivity extends Activity  

 

2.  {  

 

3.      @Override  

 

4.      public void onCreate(BundlesavedInstanceState)  

 

5.      {  

 

6.          super.onCreate(savedInstanceState);  

 

7.          setContentView(R.layout.activity_main);  

 

8.         Thread mThread = newHandlerThread("demo", Process.THREAD_PRIORITY_BACKGROUND);   

 

9.          mThread.start();  

 

10.        MyHandler mHandler = new MyHandler( mThread.getLooper( ) );  

 

11.         …….  

 

12.         …….  

 

13.         …….  

 

14.     }  

 

15.    @Override  

 

16.    public void onDestroy()  

 

17.    {  

 

18.        super.onDestroy();  

 

19.    }  

 

20.}  

 

这个代码存在泄漏问题,因为HandlerThread的run方法是一个死循环,它不会自己结束,线程的生命周期超过了activity生命周期,当横竖屏切换,HandlerThread线程的数量会随着activity重建次数的增加而增加。

 

应该在onDestroy时将线程停止掉:mThread.getLooper().quit();

 

另外,对于不是HandlerThread的线程,也应该确保activity消耗后,线程已经终止,可以这样做:在onDestroy时调用mThread.join();

 

4、注册某个对象后未注销

 

注册广播接收器、注册观察者等等,比如:

 

假设我们希望在锁屏界面(LockScreen)中,监听系统中的电话服务以获取一些信息(如信号强度等),则可以在LockScreen中定义一个PhoneStateListener的对象,同时将它注册到TelephonyManager服务中。对于LockScreen对象,当需要显示锁屏界面的时候就会创建一个LockScreen对象,而当锁屏界面消失的时候LockScreen对象就会被释放掉。但是如果在释放LockScreen对象的时候忘记取消我们之前注册的PhoneStateListener对象,则会导致LockScreen无法被GC回收。如果不断的使锁屏界面显示和消失,则最终会由于大量的LockScreen对象没有办法被回收而引起OutOfMemory,使得system_process进程挂掉。

 

5、集合中对象没清理造成的内存泄露

 

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

 

6、资源对象没关闭造成的内存泄露

 

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

 

  程序中经常会进行查询数据库的操作,但是经常会有使用完毕Cursor后没有关闭的情况。如果我们的查询结果集比较小,对内存的消耗不容易被发现,只有在长时间大量操作的情况下才会复现内存问题,这样就会给以后的测试和问题排查带来困难和风险。

 

7、一些不良代码成内存压力

 

有些代码并不造成内存泄露,但是它们或是对没使用的内存没进行有效及时的释放,或是没有有效的利用已有的对象而是频繁的申请新内存,对内存的回收和分配造成很大影响的,容易迫使虚拟机不得不给该应用进程分配更多的内存,增加vm的负担,造成不必要的内存开支。

 

7.1  Bitmap使用不当

 

   第一、及时的销毁。

 

虽然,系统能够确认Bitmap分配的内存最终会被销毁,但是由于它占用的内存过多,所以很可能会超过Java堆的限制。因此,在用完Bitmap时,要及时的recycle掉。recycle并不能确定立即就会将Bitmap释放掉,但是会给虚拟机一个暗示:“该图片可以释放了”。

 

第二、设置一定的采样率。

 

   有时候,我们要显示的区域很小,没有必要将整个图片都加载出来,而只需要记载一个缩小过的图片,这时候可以设置一定的采样率,那么就可以大大减小占用的内存。如下面的代码:

 

1.  private ImageView preview;    

 

2.  BitmapFactory.Options options = newBitmapFactory.Options();    

 

3.  options.inSampleSize = 2;//图片宽高都为原来的二分之一,即图片为原来的四分之一    

 

4. Bitmap bitmap =BitmapFactory.decodeStream(cr.openInputStream(uri), null, options); preview.setImageBitmap(bitmap);   

 

第三、巧妙的运用软引用(SoftRefrence)

 

有些时候,我们使用Bitmap后没有保留对它的引用,因此就无法调用Recycle函数。这时候巧妙的运用软引用,可以使Bitmap在内存快不足时得到有效的释放。如下:

 

1.  SoftReference<Bitmap>  bitmap_ref  = new SoftReference<Bitmap>(BitmapFactory.decodeStream(inputstream));   

 

2.  ……  

 

3.  ……  

 

4.  if (bitmap_ref .get() != null)  

 

5.       bitmap_ref.get().recycle();  

 

7.2  构造Adapter时,没有使用缓存的 convertView

 

  以构造ListView的BaseAdapter为例,在BaseAdapter中提共了方法:

 

public View getView(intposition, View convertView,ViewGroup parent)

 

  来向ListView提供每一个item所需要的view对象。初始时ListView会从BaseAdapter中根据当前的屏幕布局实例化一定数量的view对象,同时ListView会将这些view对象缓存起来。当向上滚动ListView时,原先位于最上面的list item的view对象会被回收,然后被用来构造新出现的最下面的list item。这个构造过程就是由getView()方法完成的,getView()的第二个形参 View convertView就是被缓存起来的list item的view对象(初始化时缓存中没有view对象则convertView是null)。

 

  由此可以看出,如果我们不去使用convertView,而是每次都在getView()中重新实例化一个View对象的话,即浪费时间,也造成内存垃圾,给垃圾回收增加压力,如果垃圾回收来不及的话,虚拟机将不得不给该应用进程分配更多的内存,造成不必要的内存开支。ListView回收list item的view对象的过程可以查看:

 

android.widget.AbsListView.Java--> voidaddScrapView(View scrap) 方法。

 

1.  public View getView(int position, View convertView, ViewGroupparent) {  

 

2.      View view = newXxx(...);  

 

3.      return view;  

 

4.  }  

 

  修正示例代码:

 

1.  public View getView(intposition, View convertView, ViewGroup parent) {  

 

2.  View view = null;  

 

3.  if (convertView != null){  

 

4.      view = convertView;  

 

5.      populate(view, getItem(position));  

 

6.  } else {  

 

7.      view = new Xxx(...);  

 

8.  }  

 

9.  return view;  

 

10.}  

 

7.3  不要在经常调用的方法中创建对象,尤其是忌讳在循环中创建对象。可以适当的使用 hashtable ,vector 创建一组对象容器,然后从容器中去取那些对象,而不用每次 new 之后又丢弃。

分享到:
评论

相关推荐

    使用Android Studio检测内存泄露(LeakCanary)

    内存泄露在Android开发中是一个非常重要的问题,因为它可能导致应用程序运行缓慢、消耗过多资源甚至崩溃。Android Studio提供了多种工具帮助开发者检测和解决内存泄露,其中包括Memory Profiler和LeakCanary。 ...

    Android处理内存泄漏的代码例子

    在Android开发中,内存泄漏是一个严重的问题,它会导致应用程序占用过多内存,影响性能甚至导致应用崩溃。本示例代码着重展示了如何避免Android应用中的内存泄漏,主要包括五个关键方面:关闭游标、重用适配器、回收...

    Android webview 内存泄露的解决方法

    Android webview 内存泄露的解决方法 最近在activity嵌套webview显示大量图文发现APP内存一直在涨,没法释放内存,查了很多资料,大概是webview的一个BUG,引用了activity导致内存泄漏,所以就尝试传递...

    Android Native 内存泄漏系统化解决方案

    Android Native 内存泄漏系统化解决方案 在 Android 平台上,C++ 内存泄漏问题的分析和定位一直是困扰开发人员的难题,因为地图渲染、导航等核心功能对性能要求很高,高德地图 APP 中存在大量的 C++ 代码。解决这个...

    android系统分析内存泄露测试

    在Android系统中,内存泄漏是一个严重的问题,它会导致应用程序占用过多的内存,进而影响设备性能,甚至可能导致应用崩溃。理解并有效地分析内存泄漏是每个Android开发者必须掌握的关键技能。 内存泄漏通常发生在...

    androidStudio内存泄漏分析

    **四、常见内存泄漏类型及解决方案** 1. **静态字段引用**:静态变量生命周期长,可能导致长时间持有对象。解决方案是避免在静态变量中存储非静态对象,或者使用弱引用(WeakReference)。 2. **单例模式**:不正确...

    Android内存泄漏简介

    以下是几个导致Android内存泄漏的主要因素及其相关知识点。 1. 引用没有释放造成的内存泄漏 在Android开发中,如果不正确地管理对象引用,可能会导致内存泄漏。例如,如果一个Activity持有某个服务、广播接收器或...

    android性能优化之内存泄露

    #### 二、Android内存管理基础知识 在讨论内存泄露之前,我们需要了解一些基本概念。Android应用进程通常被分配一定量的RAM(例如:2MB到由`getMemoryClass()`或`getLargeMemoryClass()`返回的值),当内存不足时,...

    Android应用程序内存泄漏的分析.pdf

    以上知识点涉及了Android内存泄漏分析的多个方面,包括理论知识、工具使用、操作实践和问题诊断技巧。理解和掌握这些知识点对Android开发人员来说是非常重要的,它可以帮助他们在日常开发中有效地避免和解决内存泄漏...

    Android解决大量帧动画内存泄漏问题

    帧动画太多造成内存泄漏问题,内含所需的JAR包:org.apache.commons.io.IOUtils,将Inputstream转为byte字节数组,减少内存泄漏 和关键代码,仅供参考,如有建议请在下方留言或添加咨询~~~

    Android 内存泄露 Mat工具分析

    Android 内存泄露是一个常见的问题,它会导致应用程序的性能下降,甚至崩溃。Memory Analyzer Tool(MAT)是一个功能强大且广泛使用的内存泄露分析工具。本文将详细介绍如何使用 MAT 分析 Android 应用程序中的内存...

    AndroidApp定位和规避内存泄露方法研究

    3. 内存泄漏的监听器:如注册了BroadcastReceiver、ContentObserver等,若忘记在合适时机注销,将导致内存泄露。 4. 异步任务:AsyncTask或其他后台任务如果引用了Activity,当Activity结束时,任务仍在后台运行,...

    android内存泄露测试

    ### Android内存泄露测试详解 在Android开发中,内存管理是一项至关重要的任务,不当的内存管理会导致应用运行缓慢、崩溃甚至耗尽系统资源。内存泄露是内存管理中的一个常见问题,它指的是应用程序分配了内存但未能...

    Android内存优化——常见内存泄露及优化方案

    2. LeakCanary:一个开源的内存泄漏检测库,能自动检测并报告内存泄露。 3. HPROF文件分析:通过生成HPROF文件,使用MAT(Memory Analyzer Tool)进行深度分析。 三、优化策略与最佳实践 1. 使用弱引用...

    基于Android Ndk/Jni的内存泄漏检测

    附件是Android下检测ndk和jni内存泄漏的demo,可以用于native中malloc和free的检测。使用方法(参见博客):https://blog.csdn.net/zhuyong006/article/details/88537499

    Android App定位和规避内存泄露方法研究

    首先,了解内存泄露的基本概念和Android内存管理模型是很有必要的。Android应用的内存主要分为几种类型:Virtual Set Size (VSS),表示虚拟耗用内存;Resident Set Size (RSS),表示实际使用的物理内存;...

    Android内存优化——常见内存泄露及优化方案.pdf

    总结来说,Android内存优化的核心在于避免内存泄露和合理分配内存资源。通过理解内存泄露的原因,运用恰当的设计模式和编程技巧,同时借助工具来监控和分析内存使用情况,可以有效地提升应用的性能,延长应用的生命...

    Node.js-Androidnative层代码内存泄漏问题调试利器

    在Android应用开发中,尤其是涉及到Native层代码时,内存泄漏是一个常见的问题,它可能导致应用程序运行缓慢,甚至崩溃。本文将详细介绍如何使用Node.js配合特定工具,有效地定位和解决Android Native层的内存泄漏...

    android检查内存泄露

    android检查内存泄露和用法 &lt;!--==============================================================================--&gt; android:name="com.squareup.leakcanary.internal.HeapAnalyzerService" android:enabled...

    Android内存优化——常见内存泄露及优化方案.zip

    常见的Android内存泄露类型包括: 1. **静态变量引用**:当一个静态变量引用了一个Activity或Context,这个对象就无法被GC回收,因为静态变量在整个应用生命周期内都存在。解决方案是避免使用静态变量持有Activity...

Global site tag (gtag.js) - Google Analytics