您的位置:首页 > 其它

如何避免Handler引起内存泄露

2015-06-08 11:06 501 查看
如果您在Activity中定义了一个内部Handler类,如下代码:

帮助
1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21
public

class
MainActivity
extends

Activity {


private

Handler mHandler =
new

Handler() {


@Override


public

void
handleMessage(Message msg) {


//TODO
handle message...


}


};


@TargetApi
(
11
)


@Override


public

void
onCreate(Bundle savedInstanceState) {


super
.onCreate(savedInstanceState);


setContentView(R.layout.activity_main);


mHandler.sendMessageDelayed(Message.obtain(),
60000
);


//just
finish this activity


finish();


}


}


然后运行Android Lint工具会有一个内存泄露警告:

This Handler class should be static or leaks might occur (com.example.ta.MainActivity.1)

Issue: Ensures that Handler classes do not hold on to a reference to an outer class

Id: HandlerLeak

In Android, Handler classes should be static or leaks might occur. Messages enqueued on the application thread’s MessageQueue also retain their target Handler. If the Handler is an inner class, its outer class will be retained as well. To avoid leaking the
outer class, declare the Handler as a static nested class with a WeakReference to its outer class.

原因是:

当Android应用启动的时候,会先创建一个应用主线程的Looper对象,Looper实现了一个简单的消息队列,一个一个的处理里面的Message对象。主线程Looper对象在整个应用生命周期中存在。

当在主线程中初始化Handler时,该Handler和Looper的消息队列关联。发送到消息队列的Message会引用发送该消息的Handler对象,这样系统可以调用 Handler#handleMessage(Message) 来分发处理该消息。

在Java中,非静态(匿名)内部类会引用外部类对象。而静态内部类不会引用外部类对象。当使用内部类(包括匿名类)来创建Handler的时候,Handler对象会隐式地持有一个外部类对象(通常是一个Activity)的引用(不然你怎么可能通过Handler来操作Activity中的View?)。、而Handler通常会伴随着一个耗时的后台线程(例如从网络拉取图片)一起出现,这个后台线程在任务执行完毕(例如图片下载完毕)之后,通过消息机制通知Handler,然后Handler把图片更新到界面。然而,如果用户在网络请求过程中关闭了Activity,正常情况下,Activity不再被使用,它就有可能在GC检查时被回收掉,但由于这时线程尚未执行完,而该线程持有Handler的引用(不然它怎么发消息给Handler?),这个Handler又持有Activity的引用,就导致该Activity无法被回收(即内存泄露),直到网络请求结束(例如图片下载完毕)。另外,如果你执行了Handler的postDelayed()方法,该方法会将你的Handler装入一个Message,并把这条Message推到MessageQueue中,那么在你设定的delay到达之前,会有一条MessageQueue
-> Message -> Handler -> Activity的链,导致你的Activity被持有引用而无法被回收。

如果外部类是Activity,则会引起Activity泄露

当Activity finish后,延时消息会继续存在主线程消息队列中1分钟,然后处理消息。而该消息引用了Activity的Handler对象,然后这个Handler又引用了这个Activity。这些引用对象会保持到该消息被处理完,这样就导致该Activity对象无法被回收,从而导致了上面说的 Activity泄露。

内存泄露的危害

只有一个,那就是虚拟机占用内存过高,导致OOM(内存溢出),程序出错。对于Android应用来说,就是你的用户打开一个Activity,使用完之后关闭它,内存泄露;又打开,又关闭,又泄露;几次之后,程序占用内存超过系统限制,FC。

方法一:将Handler声明为静态类

要修改该问题,只需要按照Lint提示的那样,把Handler类定义为静态即可,然后通过WeakReference 来保持外部的Activity对象。

静态类不持有外部类的对象,所以你的Activity可以随意被回收。代码如下:

?
1

2

3

4

5

6
static

class
MyHandler
extends

Handler {


@Override


public

void
handleMessage(Message msg) {


mImageView.setImageBitmap(mBitmap);


}


}


但其实没这么简单。使用了以上代码之后,你会发现,由于Handler不再持有外部类对象的引用,导致程序不允许你在Handler中操作Activity中的对象了。所以你需要在Handler中增加一个对Activity的弱引用(WeakReference):

?
1

2

3

4

5

6

7

8

9

10

11

12

13

14

15
static

class
MyHandler
extends

Handler {


WeakReference<Activity
> mActivityReference;


MyHandler(Activity
activity) {


mActivityReference=
new

WeakReference<Activity>(activity);


}


@Override


public

void
handleMessage(Message msg) {


final

Activity activity = mActivityReference.get();


if

(activity !=
null
)
{


mImageView.setImageBitmap(mBitmap);


}


}


}


将代码改为以上形式之后,就算完成了。

延伸:什么是WeakReference?

WeakReference弱引用,与强引用(即我们常说的引用)相对,它的特点是,GC在回收时会忽略掉弱引用,即就算有弱引用指向某对象,但只要该对象没有被强引用指向(实际上多数时候还要求没有软引用,但此处软引用的概念可以忽略),该对象就会在被GC检查到时回收掉。对于上面的代码,用户在关闭Activity之后,就算后台线程还没结束,但由于仅有一条来自Handler的弱引用指向Activity,所以GC仍然会在检查的时候把Activity回收掉。这样,内存泄露的问题就不会出现了。

转自——http://my.oschina.net/rengwuxian/blog/181449

完整参考代码如下

public class SampleActivity extends Activity {

/**
* 使用静态的内部类,不会持有当前对象的引用
*/
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);

/**
* 使用静态的内部类,不会持有当前对象的引用
*/
private static final Runnable sRunnable = new Runnable() {
@Override
public void run() { }
};

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);

//  发送一个10分钟后执行的一个消息
mHandler.postDelayed(sRunnable, 600000);

// 结束
finish();
}
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: