问题描写叙述

曾几何时,我们用原来的办法使用Handler时会有以下一段温馨的提示:

This Handler class should be static or leaks might occur

以下是更具体的说明(Android Studio上的警告,不知道Eclipse上是否同样)

Since this Handler is declared as an inner class, it may prevent the outer class from being garbage collected. If the Handler is using a Looper or MessageQueue for a thread other than the main thread, then there is no issue. If the Handler is using the Looper or MessageQueue of the main thread, you need to fix your Handler declaration, as follows: Declare the Handler as a static class; In the outer class, instantiate a WeakReference to the outer class and pass this object to your Handler when you instantiate the Handler; Make all references to members of the outer class using the WeakReference object.

大概意思就是:

一旦Handler被声明为内部类,那么可能导致它的外部类不可以被垃圾回收。假设Handler是在其它线程(我们通常成为worker thread)使用Looper或MessageQueue(消息队列)。而不是main线程(UI线程),那么就没有这个问题。

假设Handler使用Looper或MessageQueue在主线程(main thread),你须要对Handler的声明做例如以下改动:

声明Handler为static类。在外部类中实例化一个外部类的WeakReference(弱引用)而且在Handler初始化时传入这个对象给你的Handler;将全部引用的外部类成员使用WeakReference对象。

解决方式一

上面的描写叙述中基本上把推荐的改动方法明白表达了出来。以下的代码是我自己使用中的一个实现,请參考:

private CopyFileHandler mHandler;

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_appstart);
mHandler = new CopyFileHandler(this);
startCopyDBThread();
} private void startCopyFileThread(){
Log.d(TAG, "startCopyDBThread");
new Thread(new Runnable() {
@Override
public void run() {
//DO SOMETHING LIKE: copyDBFile();
Message msg=mHandler.obtainMessage();
mHandler.sendMessage(msg);
}
}).start();
} private static class CopyFileHandler extends Handler {
WeakReference<AppStartActivity> mActivity;
public CopyFileHandler(AppStartActivity activity) {
mActivity = new WeakReference<>(activity);
} public void handleMessage(Message msg) {
final AppStartActivity activity = mActivity.get();
//handle you message here!
}
}

为什么会内存泄漏

那么为什么不这样做会引发内存泄漏呢?

这与几个关键词有关:内部类、Handler的消息循环(Looper)、Java垃圾回收机制。

须要强调一下,并非每次使用Handler都会引发内存泄漏。这里面有一定的几率,须要满足特定条件才会引起泄漏。

内部类会有一个指向外部类的引用。

垃圾回收机制中约定。当内存中的一个对象的引用计数为0时。将会被回收。

Handler作为Android上的异步消息处理机制(好吧,我大多用来进行worker thread与UI线程同步),它的工作是须要Looper和MessageQueue配合的。简单的说,要维护一个循环体(Looper)处理消息队列(MessageQueue)。

每循环一次就从MessageQueue中取出一个Message。然后回调对应的消息处理函数。

假设,我是说假设,循环体中有消息未处理(Message排队中),那么Handler会一直存在。那么Handler的外部类(一般是Activity)的引用计数一直不会是0,所以那个外部类就不能被垃圾回收。

非常多人会遇到activity的onDestroy方法一直不运行就是这个原因。

还有一个解决方式的尝试

警告描写叙述中提到了Handler在worker thread中使用Looper或MessageQueue,我尝试了一下。请大家品鉴。

    private Handler testHandler;
private Thread mThread = new Thread() {
public void run() {
Log.d(TAG,"mThread run");
Looper.prepare();
testHandler = new Handler() {
public void handleMessage(Message msg) {
Log.d("TAG", "worker thread:"+Thread.currentThread().getName());
switch (msg.what) {
//handle message here
}
}
};
Looper.loop();
}
}; //start thread here
if(Thread.State.NEW == mThread.getState()) {
Log.d(TAG, "mThread name: " + mThread.getName());
mThread.start();
} //send message here
testHandler.sendEmptyMessage(1);

參考:

http://stackoverflow.com/questions/11407943/this-handler-class-should-be-static-or-leaks-might-occur-incominghandler

http://m.blog.csdn.net/blog/wurensen/41907663

http://blog.csdn.net/lmj623565791/article/details/38377229

Android实战技巧之三十八:Handler使用中可能引发的内存泄漏的更多相关文章

  1. Handler使用中可能引发的内存泄漏

    https://my.oschina.net/rengwuxian/blog/181449 http://www.jianshu.com/p/cb9b4b71a820 http://blog.csdn ...

  2. 【转】Android实战技巧之四十九:Usb通信之USB Host

    零 USB背景知识 USB是一种数据通信方式,也是一种数据总线,而且是最复杂的总线之一. 硬件上,它是用插头连接.一边是公头(plug),一边是母头(receptacle).例如,PC上的插座就是母头 ...

  3. Android实战技巧之十二:Android Studio导入第三方类库、jar包和so库

    第三方类库源码 将一网友的XMPP代码从ADT转到AS时,发现其使用了第三方类库,源码放在了lib下,直接在AS中Import project,第三方类库并没有自动导入进来,看来需要自己动手了. 项目 ...

  4. Android实战技巧之十九:android studio导出jar包(Module)并获得手机信息

    AS中并没有独立的Module 工程,可是能够在普通的Project中增加Module.所谓的Module就是我们通常所指的模块化的一个单元.并经常以jar包的形式存在.以下以一个获取手机信息的样例演 ...

  5. Android学习指南之三十八:Android手势操作编程[转]

    手势操作在我们使用智能设备的过程中奉献了不一样的体验.Android开发中必然会进行手势操作方面的编程.那么它的原理是怎样的呢?我们如何进行手势操作编程呢? 手势操作原理 首先,在Android系统中 ...

  6. Android实战技巧之三十七:图片的Base64编解码

    通经常使用Base64这样的编解码方式将二进制数据转换成可见的字符串格式,就是我们常说的大串.10块钱一串的那种,^_^. Android的android.util包下直接提供了一个功能十分完备的Ba ...

  7. (转载)Android项目实战(二十八):Zxing二维码实现及优化

    Android项目实战(二十八):Zxing二维码实现及优化   前言: 多年之前接触过zxing实现二维码,没想到今日项目中再此使用竟然使用的还是zxing,百度之,竟是如此牛的玩意. 当然,项目中 ...

  8. (转载)Android项目实战(二十八):使用Zxing实现二维码及优化实例

    Android项目实战(二十八):使用Zxing实现二维码及优化实例 作者:听着music睡 字体:[增加 减小] 类型:转载 时间:2016-11-21我要评论 这篇文章主要介绍了Android项目 ...

  9. Android实战技巧:深入解析AsyncTask

    AsyncTask的介绍及基本使用方法 关于AsyncTask的介绍和基本使用方法可以参考官方文档和Android实战技巧:多线程AsyncTask这里就不重复. AsyncTask引发的一个问题 上 ...

随机推荐

  1. 洛谷——P2681 众数

    P2681 众数 题目背景 Alice和Bob玩游戏 题目描述 Alice现在有一个序列a1.a2...an 现在她需要Bob支持询问一个区间内的众数,还要支持修改一个位置的ai 输入输出格式 输入格 ...

  2. Linux基础系列-系统密码破解

    无引导介质(光盘.iso)救援模式下root密码破解 第一步: GRUB启动画面读秒时按上下方向键,进入GRUB界面 第二步: 使用上下光标键选择要修改的操作系统启动内核(默认选择的即可),按e键进行 ...

  3. 【BZOJ 3238】【AHOI 2013】差异

    http://www.lydsy.com/JudgeOnline/problem.php?id=3238 后缀数组裸题但是\(5\times 10^5\)貌似常数有点大就过不了?(我的sa常数那么大想 ...

  4. Codeforces #447 Div.2 Tutorial

    Problem A:QAQ 给一个字符串,求出可非连续的QAQ序列有多少个. Analysis:比较水的一道题,记录每一个Q的位置,预处理A的个数即可 然而还是fst了,原因是未考虑一个Q都没有的极端 ...

  5. CDOJ 1279 班委选举 每周一题 div2 暴力

    班委选举 题目连接: http://acm.uestc.edu.cn/#/status/list?problemId=1279 Description 高考的脚步越来越近了--时间如山涧小溪一般悄无声 ...

  6. spring boot 添加自定义属性

    1.添加jar compile('org.springframework.boot:spring-boot-configuration-processor:1.2.0.RELEASE') 2.在app ...

  7. HDU 4638 Group (2013多校4 1007 离线处理+树状数组)

    Group Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submi ...

  8. Red Hat Enterprise Linux上配置SQL Server Always On Availability Group

    http://www.cnblogs.com/lavender000/p/6946848.html

  9. 使用BusyBox制作linux根文件系统(CramFS+mdev)

    转:http://www.360doc.com/content/10/0428/11/496343_25245348.shtml 操作系统:Ubuntu9.04 内核版本:linux-2.6.24.7 ...

  10. appium+python自动化50-生成定位对象模板templet(jinja2)

    前言 每次自己写pageobject定位元素对象太繁琐,格式都差不多,只是换个定位方法,这种就可以才有模板的方式,批量生成pageobject定位元素对象的模板 python里面生成模板有两个模块可以 ...