Android开发——常见的内存泄漏以及解决方案(一)
0. 前言
转载请注明出处:http://blog.csdn.net/seu_calvin/article/details/52333954
Android的内存泄漏是Android开发领域永恒的话题,那今天就总结一下常见的内存泄漏吧。也给自己提个醒,在以后的编码过程中多注意这个问题。在Android Studio里可以通过一些分析工具比如MAT来找出潜在的内存泄漏,在Android Device Monitor中对相应进行进行Dump HPROF File,并对这个文件在SDK的platform-tools目录下进行<hprof-conv infilePath outfileName>即可在platform-tools目录下生成使用MAT工具查看的hprof文件。
还有如果不清楚Java里的OOM、以及内存泄漏和内存溢出区别的小伙伴,可以参考我之前写过的Java技术——Java中的内存泄漏。
此篇将从静态变量引用Activity、匿名类、内部类、Handler、以及监听器等方面的实例说明内存泄漏的发生原因以及解决方案。
1. 单例模式导致内存泄漏(实质是静态变量引用Activity)
如果不了解单例模式的小伙伴可以查看我之前写过的设计模式——单例模式解析。已经对单例模式分析的很清楚了。这里就不多赘述了。
单例由于其静态的特性使得其生命周期跟应用一样长,处理不当极易导致内存泄漏。
public class SingleUtils {
private static SingleUtils mInstance = null;
private Context context;
private SingleUtils (Context context){
this.context = context;
} public static SingleUtils getInstance(Context context){
if(mInstance == null){
mInstance = new SingleUtils (context);
}
return mInstance;
} public Object getObject(){//根据业务逻辑传入参数
//返回业务逻辑结果,这里需要用到context
}
}
如果你看了上面链接文中介绍的单例模式,那么就很容易理解下面单例类中返回实例的代码造成了内存泄漏,我们传入了上下文context(传入上下文是为了实现单例类中的业务逻辑),这个上下文可能是Android应用中的一个Activity界面,当它finish掉的时候,这个单例类的静态对象拥有了这个Activity的引用,静态变量是驻扎在JVM的方法区,静态变量引用的对象是不会被GC回收的,因为它们所引用的对象本身就是GC ROOT。导致该Activty一直驻留在内存中,并发生内存泄漏。
解决方案:
在单例中我们尽可能的引用生命周期较长的对象,将第10行代码修改如下即可。
mInstance = new SingleUtils (context.getApplicationContext());
2. 内部类导致内存泄漏
非静态内部类会持有外部类的引用,如果我们在一个外部类中定义一个静态变量,这个静态变量是引用内部类对象,内部类能够引用外部类的成员这一优势,就是通过持有外部类的引用来实现的,但是这将会导致内存泄漏,因为这相当于间接导致静态引用外部类。
public class MyActivity extends Activity {
//非静态内部类User创建的静态实例mUser
private static InnerClass mInnerClass = null;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
mInnerClass = new InnerClass();
}
class InnerClass{
}
}
解决方案:
(1)在onDestroy方法中调用mInnerClass的判空,若不为空,手动置为null即可。
(2)将内部类定义为静态内部类,使其不能与外部类建立关系。
3.匿名导致内存泄漏
匿名内部类同样会持有一个外部类的引用,因此如果在Activity内定义了一个匿名的AsyncTask对象。如果Activity被销毁之后AsyncTask仍然在执行,那就会组织垃圾回收器回收Activity对象,进而导致内存泄漏,直到执行结束才能回收Activity。
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); new AsyncTask<Void, Void, Void>() {
@Override
protected Void doInBackground(Void... params) {
//子线程中持有Activity的引用
//子线程在Activity销毁后依然会继续执行,导致该Activity内存泄漏
while (true) ;
}
}.execute();
}
解决方案:
(1)在onDestroy中中断子线程的运行。
(2)由于线程池利于管理,可以使用全局的线程池代替在类中创建子线程。
4.Handler导致内存泄漏
定义一个匿名的Runnable对象会间接地引用定义它的Activity对象,而它会被提交到Handler的MessageQueue中,如果它在Activity销毁时还没有被处理(如下例中延迟时间执行),那就会导致内存泄漏了。
private final Handler handler = new Handler() {
@Override
public void handleMessage(Message msg) { }
}; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); handler.postDelayed(new Runnable() {
@Override
public void run() { /* ... */ }
}, Integer.MAX_VALUE);
}
解决方案:
(1)在onDestroy中清空不必要的Message消息。
(2)可以将Handler放入到静态内部类中(静态内部类不会持有外部类的引用)。如果想要在handler内部去调用所在的外部类Activity,可以在handler内部使用弱引用的方式指向所在Activity,这样不会导致内存泄漏。
5.监听器导致内存泄漏
系统服务可以通过context.getSystemService获取,它们负责执行某些后台任务,或者为硬件访问提供接口。如果 context对象想要在服务内部的事件发生时被通知,那就需要把自己注册到服务的监听器中。然而,这会让服务持有Activity的引用,如果忘记在Activity销毁时取消注册,那就会导致Activity泄漏了。
void registerListener() {
SensorManager sensorManager = (SensorManager) getSystemService(SENSOR_SERVICE);
Sensor sensor = sensorManager.getDefaultSensor(Sensor.TYPE_ALL);
sensorManager.registerListener(this, sensor, SensorManager.SENSOR_DELAY_FASTEST);
} View smButton = findViewById(R.id.sm_button);
smButton.setOnClickListener(new View.OnClickListener() {
@Override public void onClick(View v) {
registerListener();
nextActivity();
}
});
下一篇将会从资源角度来说明内存泄漏的问题。
Android开发——常见的内存泄漏以及解决方案(一)的更多相关文章
- Android开发——常见的内存泄漏以及解决方案(二)
)Android2.3以后,SoftReference不再可靠.垃圾回收期更容易回收它,不再是内存不足时才回收软引用.那么缓存机制便失去了意义.Google官方建议使用LruCache作为缓存的集合类 ...
- Android开发 |常见的内存泄漏问题及解决办法
在Android开发中,内存泄漏是比较常见的问题,有过一些Android编程经历的童鞋应该都遇到过,但为什么会出现内存泄漏呢?内存泄漏又有什么影响呢? 在Android程序开发中,当一个对象已经不需要 ...
- Android中常见的内存泄漏
为什么会产生内存泄漏? 当一个对象已经不需要再使用了,本该被回收时,而有另外一个正在使用的对象持有它的引用从而导致它不能被回收,这导致本该被回收的对象不能被回收而停留在堆内存中,这就产生了内存泄漏. ...
- android中常见的内存泄漏和解决的方法
android中的内存溢出预计大多数人在写代码的时候都出现过,事实上突然认为工作一年和工作三年的差别是什么呢.事实上干的工作或许都一样,产品汪看到的结果也都一样,那差别就是速度和质量了. 写在前面的一 ...
- Android性能优化之常见的内存泄漏
前言 对于内存泄漏,我想大家在开发中肯定都遇到过,只不过内存泄漏对我们来说并不是可见的,因为它是在堆中活动,而要想检测程序中是否有内存泄漏的产生,通常我们可以借助LeakCanary.MAT等工具来检 ...
- 老李分享:Android性能优化之内存泄漏1
老李分享:Android性能优化之内存泄漏 前言 对于内存泄漏,我想大家在开发中肯定都遇到过,只不过内存泄漏对我们来说并不是可见的,因为它是在堆中活动,而要想检测程序中是否有内存泄漏的产生,通常我 ...
- 5个Android开发中比较常见的内存泄漏问题及解决办法
android中一个对象已经不需要了,但是其他对象还持有他的引用,导致他不能回收,导致这个对象暂存在内存中,这样内存泄漏就出现了. 内存泄漏出现多了,会是应用占用过多的没存,当占用的内存超过了系统 ...
- Android开发常见的Activity中内存泄漏及解决办法
上一篇文章楼主提到由Context引发的内存泄漏,在这一篇文章里,我们来谈谈Android开发中常见的Activity内存泄漏及解决办法.本文将会以“为什么”“怎么解决”的方式来介绍这几种内存泄漏. ...
- JavaScript如何工作:内存管理+如何处理4个常见的内存泄漏
摘要: 作者将自己常用的JavaScript模块分享给大家. 原文:JavaScript如何工作:内存管理+如何处理4个常见的内存泄漏 作者:前端小智 Fundebug经授权转载,版权归原作者所有. ...
随机推荐
- Get和Post的请求
get post请求 <form method="post","get", action="a.ashx"> <input ...
- 我对USB的认识
一.USB协议规范 (1) 基本概念 每一个设备(device)会有一个或者多个的逻辑连接点在里面,每个连接点叫endpoint.每个endpoint有四种数据传送方式:控制(Contr ...
- spring boot 启动报错No active profile set, falling back to default profiles
报错No active profile set, falling back to default profiles pom.xml加上下面两个依赖 <dependency> <gro ...
- Java基础反射-调用类
Student类 package com.test.wang; import java.lang.reflect.Constructor; import java.lang.reflect.Field ...
- 金三银四面试季节之Java 核心面试技术点 - JVM 小结
原文:https://github.com/linsheng9731/notebook/blob/master/java/JVM.md 描述一下 JVM 的内存区域 程序计数器(PC,Program ...
- Erlang 001--开篇
有段时间没有更新博客了,最近稍微接触了下一门相对小众的语言Erlang,个人感觉学习一段时间有必要总结总结,本文作为该系列的开篇,仅仅列举一些与Java的一些不同点和个人对Erlang的一些主观印象, ...
- 使用JAVA读写Properties属性文件
使用JAVA读写Properties属性文件 Properties属性文件在JAVA应用程序中是经常可以看得见的,也是特别重要的一类文件.它用来配置应用程序的一些信息,不过这些信息一般都是比较少的数 ...
- 【ros-kinetic iai_kinect2 opencv2 3 】注意事项
iai_kinect2 : https://github.com/code-iai/iai_kinect2/tree/master/kinect2_registration kinect2_brid ...
- RTMP,RTMPT,RTMPS,RTMPE,RTMPTE协议的介绍
1. AMF AMF(是Action Message Format的缩写)是在flash和flex中与远程服务端交换数据的一种格式.它是二进制格式,Flash应用与服务端或数据库通过RPC交换数据时, ...
- [习题] FindControl 简单练习--GridView + CheckBox,点选多列数据(复选删除)#3 List或数组
[习题] FindControl 简单练习--GridView + CheckBox,点选多列数据(复选删除)#3 List或数组 之前的范例,使用字符串.文字来记录将删除的文章ID 后续会有很多小缺 ...