一、背景和目的:

目前许多开发人员在Android开发过程中,较少关注实现细节和内存使用,容易会造成内存泄露,导致程序OOM。

本文会通过代码向大家介绍在Android开发过程中常见的内存泄露。

二、常见的内存泄露代码

1、使用Handler造成的内存问题

在Android开发过程中,Handler是比较常用的,通过Handler发送Message与主线程进行通信,Message发送之后是存储在MessageQueue中的,有些Message并不是马上被处理的,在Message中存在一个Target,是Handler的一个引用,如果Message在Handler中的存在时间过长,会导致Handler无法被回收。如果Handler非静态,则会导致相关引用的Activity或者Service不会回收,所以在处理Hanlder之类的内部类的时候,应该要将Handler定义为静态内部类,同样在使用HandlerThread的时候也需要注意,我们来看看代码:

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

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

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

2、使用非静态内部类的静态实例

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

3、在Activity中使用静态成员

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

以上2个例子的内存泄漏都是因为Activity的引用的生命周期超越了activity对象的生命周期。也就是常说的Context泄漏,想要避免context相关的内存泄漏,需要注意以下几点:

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

l在可以使用application的context的情况下,尽可能使用application的context来替代和activity相关的context

l如果一个acitivity的非静态内部类的生命周期不受控制,那么我们就应该避免这样使用。

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

注册广播接收器、注册观察者等等,比如: 在调用registerReceiver后,若未调用unregisterReceiver,它会导致BroadcastReceiver不会被unregister而导致内存泄露,我们经常会看到类似下面的代码:

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

我们通常把一些对象的引用加入到了集合中,当我们不需要该对象时,如果没有把它的引用从集合中清理掉,这样这个集合就会越来越大。如果这个集合是static的话,如果对象不断增大,达到一定的值的时候程序就会OOM

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

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

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

写代码时,经常会有人忘记调用close, 或者因为代码逻辑问题状况导致close未被调用。

错误的代码:

修正后的代码:

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

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

7.1Bitmap使用不当

一、需要及时的销毁。

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

二、需要设置一定的采样率。

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

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

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

7.2使用Adapter时,没有使用缓存的 ConvertView

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

来向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

错误的代码:

修正示例代码:

7.3适当的使用对象池

不要在经常调用的方法中创建对象,每次new之后都丢弃,尤其是忌讳在循环中创建对象。在android support v4包中包含Pools类,其实就是对象池,使用方法也比较简单,具体可以参考下面的MyPools这个类。

百度MTC是业界领先的移动应用测试服务平台,为广大开发者在移动应用测试中面临的成本、技术和效率问题提供解决方案。同时分享行业领先的百度技术,作者来自百度员工和业界领袖等。

android如何避免内存泄露,Android开发中应该避免的内存泄露相关推荐

  1. 5 个 Android 开发中比较常见的内存泄漏问题及解决办法

    Android开发中,内存泄漏是比较常见的问题,有过一些Android编程经历的童鞋应该都遇到过,但为什么会出现内存泄漏呢?内存泄漏又有什么影响呢? 在Android程序开发中,当一个对象已经不需要再 ...

  2. Android事件分发机制在实战开发中的应用之一

    学习的最终目标就是要学以致用,本文所分享的案例都是自己在公司实战开发过程中的真实案例,现在把它分享出来,希望对初学者有所帮助 版权声明:本文来自门心叼龙的博客,属于原创内容,转载请注明出处:https ...

  3. Android事件分发机制在实战开发中的应用之二

    学习的最终目标就是要学以致用,本文所分享的案例都是自己在公司实战开发过程中的真实案例,现在把它分享出来,希望对初学者有所帮助 版权声明:本文来自门心叼龙的博客,属于原创内容,转载请注明出处:https ...

  4. android开发中,可能会导致内存泄露的问题

    转自 : http://spencer-dev.lofter.com/post/d7b9e_6faf120 在android编码中,会有一些简便的写法和编码习惯,会导致我们的代码有很多内存泄露的问题. ...

  5. linux内存使用统计,Linux 中free命令检查内存使用情况

    我们都知道, IT 基础设施方面的大多数服务器(包括世界顶级的超级计算机)都运行在 Linux 平台上,因为和其他操作系统相比, Linux 更加灵活.有的操作系统对于一些微乎其微的改动和补丁更新都需 ...

  6. android引用的java包_Android开发中jar包的创建及引用

    jar包有两种,一种是纯粹的库功能,没有main入口函数,故无法直接执行:一种是包含main的可执行包! 如有hello/hello.java文件: $ cd hello/$ javac hello. ...

  7. android 开启一个定时线程_ANDROID开发中定时器的3种方法

    在android中,经常用到的定时器主要有以下几种实现: 一.采用Handler与线程的sleep(long )方法 二.采用Handler的postDelayed(Runnable, long) 方 ...

  8. Android碎片知识(十).开发中的gps定位(转)

    一.LocationManager LocationMangager,位置管理器.要想操作定位相关设备,必须先定义个LocationManager.我们可以通过如下代码创建LocationManger ...

  9. 【Android 逆向】代码调试器开发 ( ptrace 函数 | 向进程内存写出数据 )

    文章目录 一.向进程内存写出数据 二.写出流程 三.完整代码 一.向进程内存写出数据 向内存写出数据 : 每次最多能写出 4 字节 ; ptrace(PTRACE_POKETEXT, m_nPid, ...

最新文章

  1. Python培训中有哪些是必须学的运算符
  2. 使用JUnit进行单元测试
  3. 异常处理器详解 Java多线程异常处理机制 多线程中篇(四)
  4. DDD领域驱动之干货(二)
  5. matlab treeview,treeview控件
  6. 矩阵分解 java_使用矩阵分解为推荐系统
  7. python实例 97,98
  8. 操作系统——实验贰——进程通信(一)管道及共享内存
  9. javascript sort排序
  10. WCF分布式开发常见错误(13):此方法调用的事务被异步中断
  11. Linux那些事儿----好的学习网站
  12. ArduinoUNO实战-第十一章-4位共阴数码管实验
  13. 简单高效!25个漂亮的简约风格网站设计作品
  14. 使用NLTK实现stemming
  15. 生产线平衡优化毕业论文【flexsim仿真】
  16. 科研画图都用什么软件
  17. ArcGIS 图层上点的编辑
  18. 看完这一篇,你也可以自如地掌握字节码插桩
  19. 关于兼容北斗版本808协议中位置信息汇报0x0200的解析
  20. PS Suite Studio 初探

热门文章

  1. 运用单例模式、建造者模式和策略模式实现异步加载Android联系人资料
  2. 宠物龟 扫地机器人_有宠物家庭必选 岚豹扫地机器人太实用了
  3. 计算机应用基础 a)卷,《计算机应用基础》(A卷)44648
  4. 【SpringCloud】Eureka工作原理
  5. paddleoc onnx转换推理
  6. DiracNetV2
  7. windows msys编译64位x264和ffmpeg
  8. pyhton object is not subscriptable 解决
  9. java树 JTree实例(可动态添加节点)
  10. 53 Paramiko的使用