老生常谈:Handler消息机制进阶

说明:本内容为《安卓开发艺术探索》第10章内容.

   写在前面:从开发的角度讲,Handler是Android消息机制的上层接口.通过handler机制可以轻松地将
一个任务切换到Handler所在的线程中去执行.很多人认为Handler的作用是更新UI,没错但是更新UI仅仅
是Handler的一个特殊使用场景.
   具体来说是这样的:有时候需要在子线程中进行耗时的I/O操作,可能是读取文件或者访问网络,当操作完
成时需要在UI上做一些改变,由于Android开发规范限制,不能再子线程更新UI,我们就可以通过Handler
就可以将更新Ui的操作切换到主线程执行.因此,本质上来说Handler并不是专门为了用于更新UI的,只是常
常被开发者用来更新UI.
   Handler的运行需要底层的 MessageQueue 和 Looper 的支撑。

   MessageQueue是一个消息队列,内部存储了一组消息,以队列的形式对外提供插入和删除的工作,内
部采用单链表的数据结构来存储消息列表。
   Lopper在这里可以理解为消息循环,由于MessageQueue只是一个消息的存储单元,不能去处理消息,
Looper就填补了这个功能.会以无限循环的形式去查找是否有新消息,如果有就处理消息,否则就一直
等待着。(Looper中还有一个特殊的概念叫ThreadLocal,它并不是线程,它作用是在每个线程中存储数
据.ThreadLocal可以在不通的线程中互不干扰的存储并提供数据,Handler通过ThreadLocal可以获取
每个线程中的Looper。线程是默认没有Looper的,使用Handler就必须为线程创建Looper。我们经常
提到的主线程,也叫UI线程,它就是ActivityThread,被创建时就会初始化Looper。
1.为什么子线程不能更新UI?
   如果子线程访问UI,程序会抛出线程错误异常;ViewRootImpl在checkThread()中做了判断。由
于Android不建议在主线程进行耗时操作,否则可能会导致ANR。那我们耗时操作在子线程执行完毕
后,我们需要将一些更新UI的操作切换到主线程当中去。所以系统就提供了Handler。
   系统为什么不允许在子线程中去访问UI呢? 因为Android的UI控件不是线程安全的,多线程并
发访问可能会导致UI控件处于不可预期的状态,有人要问为什么不对UI的访问加上锁机制呢?因为这
样会让UI访问逻辑变得复杂;其次锁机制会降低UI访问的效率,因为锁机制会阻塞某些线程的执行。
所以Android采用了高效的单线程模型来处理UI操作。

2.Handler原理:
Handler创建时会采用当前线程的Looper来构建内部的消息循环系统,如果当前线程没有
Looper就会报错。Handler可以通过post方法发送一个Runnable到消息队列中,也可以通
过send方法发送一个消息到消息队列中,其实post方法最终也是通过send方法来完成。
send()的工作过程: 它会调用MessageQueue的enqueueMessage方法最终将这个消息放到消息队
列中,当Looper发现有新消息到来时,处理这个消息,最终消息中的Runnable或者Handler的
handleMessage方法就会被调用,注意Looper是运行Handler所在的线程中的,这样一来
业务逻辑就切换到了Handler所在的线程中去执行了。
3.Android的消息机制分析:

3.1ThreadLocal的工作原理
一般的开发中,我们很少使用到ThreadLocal.
ThreadLocal是一个线程内部的数据存储类, 通过他可以在指定的线程中存储数据. 数据存储以后, 
只能在指定线程中可以获取到存储的数据. 而其他线程无法获取.
而Android中的Looper ,ActivityThread, AMS都用到了ThreadLocal. 可以这样来说: 当某
些数据是以线程为作用域并且不同线程具有不同的数据副本的时候, 可以采用ThreadLocal.

小例子:定义一个mLocalThread对象,在主线程和两个子线程中同时操作这个对象,发现在获取值的时候却不
相同. 这是因为: 不同的线程访问同一个ThreadLocal#get()方法的时候, ThreadLocal内部会从各自的
线程中取出一个数组, 然后再从数组中根据当前ThreadLocal的索引去查找出对应的value值. 可以看出
不同的线程中的数组是不相同的. 所以也就是为什么使用ThreadLocal可以在不同的线程中维护一套数据的
副本并且彼此互不影响.

3.2消息队列的工作原理
消息队列在android中指的是MessageQueue, MessageQueue主要包含两个操作: 插入和读取. 读取操作本
身会伴随着删除的操作, 插入和读取对应的方法分别为enqueueMessage(),next(). 前者就是往消息队列中
插入一条消息, 而后者就是取出一条消息并将其从消息队列中移除. 上面说过虽然MessageQueue称为消息队
列, 但是内部实现使用的是单链表的数据结构来维护的消息列表. 单链表在插入和删除上比较有优势.

3.3Looper的工作原理
Looper在Android的消息机制中扮演者消息循环的角色, 具体来说就是他会不停地从MessageQueue中查看
是否有新消息. 如果有新消息就会处理. 否则就一直阻塞在那里. 先从构造方法开始, 在构造方法中他会
创建一个MessageQueue即消息队列, 然后将当前线程的对象保存起来.
looper的退出:
-- quit(): 这个方法会直接退出Looper
-- quitSafely(): 设定一个退出标记, 然后把消息队列中的已有消息处理完毕后才安全的退出.
如果Looper退出, 通过Handler发送的消息会失败, 这个时候Handler发送的消息会失败, 而
Handler#send()方法这个时候回返回false. 我们建议,在子线程中, 如果手动为其创建了Looper, 
那么在所有的事情完成以后应该调用quit()方法来终止消息循环. 否则这个线程会一直处于等待的状态,
 而如果退出了Looper以后, 这个线程就会立刻终止.
Looper最重要的一个方法loop()方法, 只有调用了loop后, 消息循环系统才会真正的起作用:
首先这个loop()方法是一个死循环, 唯一跳出循环的方式就是MessageQueue#next()方法返回null.
Looper必须退出, 否则loop方法就会无限循环下去. loop()会调用MessageQueue#next()方法来获取
新消息. 而next是一个阻塞操作, 当没有消息时, next方法就会一直阻塞在那里. 这也导致loop()会一直
阻塞在那里. 如果MessageQueue#next()返回了新消息, Looper就会处理这条消息: msg.target
.dispatchMessage(msg), 这里的msg.target是发送这条消息的Handler对象, 这样Handler发送的消息最
终又交给它的dispatcherMessage()来处理. 但是这里不同的是, Handler#dispatcherMessage()方法是
在创建Handler时所使用的Looper中执行的. 这样就成功的将代码逻辑切换到指定的线程中去执行了.
4.Handler工作原理:
Handler主要包含消息的发送和接收过程. 消息的发送可以通过post(或send)方法来实现的.
Handler发送消息的过程仅仅是向消息队列中插入了一条消息, MessageQueue#next()方法就是返回这条消
息给Looper, Looper收到消息后就开始处理. 最终消息有Looper交由Handler处理, 即Handler#dispatch
Message()方法会被调用, 这个时候Handler就会进入了处理消息的阶段.
处理消息的过程:
先检查msg.callback属性是否为null, 不为null就通过handleCallback()来处理消息. msg.callback是
一个Runnable接口, 实际上就是post()中传递的Runnable参数.
其次检查mCallback是否为null, 不为null就调用mCallback.handleMessage(msg)方法来处理消息.
 Callback是一个接口, 通过Callback可以采用如下方式来创建Handler对象: Handler handler =
 new Handler(callback). 通过源码注释了解: 这个接口可以用来创建一个Handler的实例但并不需要
派生Handler的子类并重写其handleMessage方法来处理具体的消息, 而CallBack给我们提供了另外一种
方式使用Handler. 当我们不想派生子类时, 就可以通过Callback来实现.

最后, 调用Handler#handleMessage()方法来处理消息.

Handler还有一个特殊的构造方法, 那就是通过一个特定的Looper来构造Handler,最常用的就是直接new
 出一个Handler, 这个构造方法会调用下面的构造函数. 很明显这就是为什么当前线程没有Looper的话, 
就会抛出Can't create handler inside thread that has not called Looper.prepare()这个异常.
5.主线程的消息循环
Android的主线程就是ActivityThread, 主程序的入口方法为main(), 在main()中系统通过Looper.prepareMainLooper()
来创建主线程的Looper以及MessageQueue, 并通过Looper.loop()来开启主线程的消息循环.

当主线程的消息循环开始以后, ActivityThread还需要一个Handler来和消息队列进行交互, 这个Handler
就是ActivityThread.H, 它的内部定义了一组消息类型, 主要包含了四大组件的启动和停止等过程.
例如:
   public static final int LAUNCH_ACTIVITY       = 100;
   public static final int PAUSE_ACTIVITY          = 101;
ActivityThread通过ApplicationThread和AMS进行进程间通信, AMS以进程间通信的方式完成ActivityThread
的请求后回调ApplicationThread中的Binder()方法, 然后Application会向H发送消息, H收到消息后会将
ApplicationThread中的逻辑切换到ActivityThread中去执行, 即切换到主线程去执行, 这个过程就是主线
程的消息循环模型.
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,684评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,143评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,214评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,788评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,796评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,665评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,027评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,679评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,346评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,664评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,766评论 1 331
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,412评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,015评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,974评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,073评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,501评论 2 343

推荐阅读更多精彩内容