【Android】简单全面的理解Handler机制

前言:Handler机制应该是网上讲解最多的一种机制(没有之一),本篇用通俗易懂的语言来介绍一下Handler机制,让大家可以更好的理解。

什么是Handler机制?

Handler机制是AndroidSDK提供的一个非常重要的处理异步消息的机制,主要是由Handler、Looper、Message和MessageQueue组成,Handler只是消息处理机制的一部分。

  • Message:消息(分为硬件产生的消息和软件产生的消息)。
  • MessageQueue:消息队列,主要是向消息池投递消息(MessageQueue.enqueueMessage)和取走消息池中的消息(MessageQueue.next)。
  • Handler:主要功能是向消息池发送消息(Handler.sendMessage)和处理消息(Handler.handleMessage)。
  • Looper:不停的循环执行(Looper.loop),从MessageQueue中取出Message并发送给Handler。
分析上述各部分:

Message:什么是硬件消息和软件消息呢?硬件消息就是我们滑动触摸点击按钮等等,软件消息就是我们主动new Message发送出去的。Message实现了Parcelable接口封装消息数据,所以他是存在于内存中的。一个实体(类)如果需要封装到消息中去就必须实现这一接口。
MessageQueue:相当于一个容器,消息池。上述看到了.next应该猜到是链表形式,实际上确实是单链表维护,在插入和删除上有优势。在其next()中会无限循环,不断的判断是否有消息,有就返回这条消息并移除。
Looper:Looper创建的时候会创建一个MessageQueue,它们两个是一一对应的关系。调用Looper.loop()的时候消息循环开始,不断地调用MessageQueue的next()方法,当有消息就处理,否则就堵塞在next()方法中。loop()跟MessageQueue的next()一样都是死循环(源码可见for(;;))。退出时调用Looper.quit(),它会调用MessageQueue的quit()方法,此时next会返回null,然后loop()方法也跟着退出。
Handler:在主线程构造Handler,new Handler()里调用了Looper.myLooper()这个方法,这个方法是获取当前线程的Looper的。在其他线程调用sendMessage()时主线程的MessageQueue会插入一条Message,然后被Looper使用,在Looper的loop()中通过回调 msg.target.dispatchMessage(msg);发送给Handler。Handler跟Looper的关系是多对一。

一段Looper的源码片段:

解释完各部分的分工那来总结一下他们之间的关系:Handler负责发送消息(Message)到MessageQueue中,Looper负责循环的接收MessageQueue中的消息通过回调方法返还给Handler自己本身。

Handler机制流程图

容易忽略的Message,使用时应注意的地方有哪些?

Message在Handler机制中看似很不起眼,但至关重要。它封装了任务携带的信息和该任务的handler,使用时应注意:

  • Message可以通过 new 来获取,但通常使用Message.obtain()或Handler.obtainMessage()方法来从消息池中获取空消息对象,可以节省资源!
  • 如果Message只需要携带简单的int型数据,优先使用arg1和arg2来传递数据,比Bundle节省内存。
  • 使用Message.what来标识信息便于处理Message。
  • 最后如果需要从工作线程返回很多数据信息再借助Bundle对象将数据集中到一起放在obj属性中,返回到主线程处理。

Looper源码简述:

上面说了那么多Looper的方法,但在Activity中使用Handler时没看到过Looper的影子啊,原来Activity内部包含了一个Looper对象,它会自动管理Looper并处理子线程中发送过来的消息。前面说到过,初始化Handler时在Handler的构造函数中会把当前线程的Looper与Handler关联,所以在Activity中无需显式的使用Looper。

但在子线程中则需要我们自己维护Looper。
源码简述

网上介绍Handler机制的文章解释最详细的就是Looper,会附带源码每一篇都会解释的很透彻。这里就简单的总结一下Looper,详细请去查阅Looper源码。

可以看出Prepare()的工作方式核心就是将Looperd对象定义为ThreadLocal,将唯一的Looper对象添加到ThreadLocal中。
Looper在构造方法中创建了一个MessageQueue和当前线程Thread。
那什么是TheadLocal呢?
ThrealLocal 是一个泛型类。
工作原理:ThreadLocal存储数据时先获取当前线程,然后通过当前线程来创建Values,如果没有Values就new一个Values实例。然后存储传进来的Value。获取的时候也是根据当前线程的Values来获取的。
它的get()和set()方法如下图:
使用场景:数据是以线程为作用域并且不同线程具有不同的数据副本时可以使用。

最后说一下为什么Handler机制会造成内存泄露

因为非静态内部类导致的!
我们知道非静态内部类默认就会持有外部类的引用,当非静态内部类对象的生命周期比外部类还长就会导致内存泄露。


上面介绍Message时说过mHandler会作为成员变量保存在发送的消息msg中,所以msg就会持有mHandler的引用,而mHandle是MainActivity的非静态内部类实例,所以mHandler就持有MainActivity的引用。msg间接持有MainActivity的引用。msg发送到消息队列(MessageQueue)中等待Looper轮询处理。当MainActivity退出后,msg可能还存在消息队列中未处理或正在处理。这样就会导致MainActivity无法被回收,以致发生MainActivity的内存泄露。
解决办法:使用静态内部类+弱引用的方式
mHandler通过弱引用的方式持有MainActivity,当GC执行垃圾回收时遇到MainActivity就会回收并释放所占的内存单元,避免内存泄露。msg还可能存在MessageQueue中,所以在MainActivity销毁时将mHandler的回调和发送的消息给移除掉。

结束

到此Handler机制的内容就算讲完了,但本篇只是用比较容易理解的方式介绍了一下Handler机制,Handler机制在Android开发中随处可见也很重要。大家很有必要去深入研究一下,去看一下里面的源码或结合一些讲解源码的文章自己理解一下。希望能帮到大家,有错的地方请提出来哈。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 195,898评论 5 462
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 82,401评论 2 373
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 143,058评论 0 325
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,539评论 1 267
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,382评论 5 358
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,319评论 1 273
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,706评论 3 386
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,370评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,664评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,715评论 2 312
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,476评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,326评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,730评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,003评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,275评论 1 251
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,683评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,877评论 2 335

推荐阅读更多精彩内容