Android内存泄漏问题,其实在开发中遇到的不多,基本都是细节问题导致泄漏,但是也必须要了解它并且解决他,提高代码质量,先分析java中的内存泄漏问题加强基础,再分析Android中的内存泄漏问题(Android中的内存泄漏问题平时开发注意即可,主要还是代码习惯问题,这里主要了解原因及注意事项)。

java中的内存泄漏

简单的说在java中内存泄漏,就是应该被释放的对象没有释放,被某个或多个实例所持有,但是不再被使用导致GC也无法回收。(原理部分所涉及堆栈的东西不做记录,了解原因即可)


java是如何进行内存管理

首先java内存主要分为三部分:静态存储区、栈区、堆区(需要注意的点是这里的堆栈和数据结构中的堆栈概念不一样,不要混在一起了,这里指内存空间)

静态方法区:主要存储静态数据即static修饰的数据和常量。这里的内存在程序编译时已经分配好了,在程序运行的整个生命周期都是存在的。

栈区:主要存放局部变量(这里指方法执行时的基本数据类型,和创建对象的引用,方法执行结束后,这些局部变量所占有的内存空间将自动释放,执行下个方法时,又会存放其他方法的局部变量。)

堆区:主要存放类的成员变量(这里的成员变量是因为类被new出来后即存入堆内存所以类的成员变量也属于堆内存)和new出来的对象及数组,这里是垃圾回收器自动管理的,通过地址可以访问数据。

  • 对java的内存分配了解后,我们就可以继续看看java是如何管理内存的:
    java的内存管理的本质其实就是对象的分配与释放,我们广大程序员去new对象,为对象申请内存空间,当然基本数据类型是除外的。我们new出来的对象申请的空间是在Heap中分配的。而对象的释放则是GC来完成的。简单总结下就是程序员通过程序代码来申请内存空间,并在Heap中分配空间,而释放空件则是JVM通过GC来完成的。GC释放对象回收内存的基本前提是该对象没有任何引用,也不会再被引用。

GC的工作方式是有向图的形式,这样虽然效率低些,但是可以很好的处理对象循环引用问题。比如当三个对象循环引用时,但是没有指向根顶点时我们可以将他们视为无效的对象,GC会回收他们。当然我们给对象赋予空值null时或者给对象赋予了新值,那么这部分对象将被回收。


java中的内存泄漏问题

java虚拟机的垃圾回收机制是通过有向图的形式,回收那些没有可达路线的对象。但是当这个对象是可达的,但是这个对象我们以后再也不会使用了,而引用这个对象的实例一直存在内存里面,那么可以判断这里内存泄漏了。内存泄漏可能影响并不会很严重,但是有时候会出现out of memory的情况发生。所以我们要时刻注意这种情况的发生,并加以避免。(虽然GC给出了几个函数调用如System.gc()但是由于虚拟机可能不是同一个虚拟机,也可能为了实现某种需求,进行了算法的调整,所以这些函数并不一定会执行)

  • 典型的java内存泄漏问题
    1.静态集合类引起内存泄漏:

静态的HashMap或者ArrayList等集合类最容易发生举个列子:

Static ArrayList ar = new ArrayList();for (int i = 1; i<10; i++){Object o = new Object();ar.add(o);o = null;}

这里的o虽然被赋值为了null,但是在集合ar中依旧有着o的引用而且由于是静态变量所以会一直持有引用导致GC无法释放,这里就发生了内存泄漏。

2.集合中的对象属性被修改后导致内存泄漏:
我们假设new了A对象,并命名为a并添加到HashSet中,那么这时候我们修改a的属性值,这时候a的HashCode会发生变化,这时候,HashSet中依旧存储这老的HashCode,所以这时候,就出现了内存泄漏问题。(不过我在测试过程中发现并没有内存泄漏,我测试使用的是jdk1.8,还没发现这里的矛盾怎么解释)

3.数据库连接io连接及网络连接等;
注意需要显示的close掉,不然可能出现内存泄漏

4.内部类和外部模块的引用
跟标题内容类似,如果创建了静态的内部类实例,由于内部类持有外部类引用,所以可能会导致内存泄漏

5.单例模式
因为单列是静态实例,所以很有可能持有其他类的引用无法释放,注意使用即可。


Android中的内存泄漏问题

下面我主要例举多个容易发生内存泄漏的情况,及每种情况该如何解决:

  1. 上下文Context的使用导致内存泄漏
    原因:这里导致内存泄漏的原因是当一个长生命周期的实例(以单列为例)中引用的Context而这个Context只是一个Activity的Context,这时候关闭这个Activity时,由于Context被这个单列所引用,所以这个Activity实例无法释放,导致内存泄漏。
    解决:在需要传递Context作为参数时,尽量传递Application的context,因为Application的Context存在整个生命周期,不会出现内存泄漏的情况,需要注意的事,有的地方比如弹出对话框之类的api时需要使用本Activity的Context注意使用场景即可。

2.Handler 造成的内存泄漏
原因:Handler、Message 和 MessageQueue 都是相互关联在一起的,当 Handler 发送的 Message 尚未被处理,则该 Message 及发送它的 Handler 对象将被线程 MessageQueue 一直持有。因为Handler的生命周期和Activity,Service等不一致,所以可能Activity可能无法正常销毁,可能导致内存泄漏,同时当Activity销毁时消息也可能还没有完全处理掉。
解决:设置Activity为弱引用可以避免Activity的内存泄漏,设置Handler为静态内部类使其生命周期和应用相等,同时在Activity的onStop或者onDestroy方法中进行消息的清空避免消息未处理的情况发生。

3.其他资源未关闭导致内存泄漏
解决方法:销毁Activity时,关闭资源的使用

4.再有其他的情况主要就是进行java程序开发时,没有注意发生的,属于java内存泄漏的典型问题。

更多相关文章

  1. Android产品研发(二十三)-->Android中保存静态秘钥实践
  2. Android 应用进程保活APP常驻内存研究方案
  3. android内存及内存溢出分析详解
  4. Android 之 内存管理
  5. android的内存监控
  6. android 图片内存溢出
  7. Android解析XML到对象工具类
  8. android 字符串转换成JSON对象

随机推荐

  1. Android反编译和二次打包实战
  2. Android无线调试――抛开USB数据线
  3. Android(安卓)面试必备 - http 与 https
  4. android 简单地设置Activity界面的跳转动
  5. unity, imageEffect在android上不显示的
  6. 《第一行代码》读完总结
  7. 将JavaFX运行到Android上
  8. android canvas常用的方法解析(一)
  9. 隐形成本巨大 微软点评Android六宗罪
  10. Android应用程序私有目录下文件操作总结