2018-04-12—(重点)源码角度分析Handler运行原理

今天给大家讲一个比较难搞得东西,Handler,其实handler用起来比较方便,流程也不复杂,但是如果我们深入源码的话,就比较头大了,今天我们来 一起从源码角度分析一下handler的工作原理:



一、Handler使用   


首先我们要知道Handler是干什么的?

handler主要是用于子线程和主线程之间进行数据交互(通信)。当需要有耗时操作(HTTP请求,数据库访问等),我们在子线程完成这些操作,而完成之后通过handler来进行 通信,下面我们来看一下handler的使用:


1.主线程使用handler

在MainActivity中创建一个handler对象。

2018-04-12—(重点)源码角度分析Handler运行原理
MainActivity新建handler对象

写一个按钮

2018-04-12—(重点)源码角度分析Handler运行原理
写一个按钮,定义点击事件的方法名字

在上面定义的but方法中,用handler发送一条信息,

2018-04-12—(重点)源码角度分析Handler运行原理
点击事件方法中发送一个message

然后我们回到handler中,接收这个message

2018-04-12—(重点)源码角度分析Handler运行原理
当有message时候,如果what是0,的执行逻辑

好了,我们来看一下他的效果

2018-04-12—(重点)源码角度分析Handler运行原理
效果

这是在主线程中使用handler,下面我们来看一下在子线程中是怎么样使用的:


2.在子线程中使用handler

定义个新的按钮,

2018-04-12—(重点)源码角度分析Handler运行原理
第二个按钮,

在点击事件中创建一个新的线程,并且在线程里面对一个handlerThread初始化。

2018-04-12—(重点)源码角度分析Handler运行原理
第二个按钮的点击事件

大家注意到没,在子线程中,多了两个方法Looper.prepare()和Looper.loop();两个方法。这个如果在子线程中使用Handler必须要写这两个方法,至于原因我们等一会儿再说。

然后我们分别用handlerThread和handler发送一条消息,

2018-04-12—(重点)源码角度分析Handler运行原理

为了看的明显,我们延时了3秒。

接着我们写handlerThread中的handleMessage方法。

2018-04-12—(重点)源码角度分析Handler运行原理
handlerThread的handleMessage方法

好了,我们来跑一下,看看效果:

2018-04-12—(重点)源码角度分析Handler运行原理
效果2

好了, 这是handler的两种用法,我们已经学会了。




二、从Handler源码分析Handler工作原理

大家还记得刚才我们在子线程多加的两个方法吗?Looper.prepare()和Looper.loop();这两个方法是必须要执行的,因为不是我们主线程不用执行,而是主线程已经自动帮我们执行了这两个方法了,我们进入main方法看一下:


2018-04-12—(重点)源码角度分析Handler运行原理
main方法
2018-04-12—(重点)源码角度分析Handler运行原理
main方法

我们注意到绿色方框框起来的两个方法正是Looper.prepare()和Looper.loop()。

那么这两个方法到底是干嘛的呢?不要着急,我们首先来看一下Handler的工作流程:


2018-04-12—(重点)源码角度分析Handler运行原理
Handler流程图

这是我们Handler的大概流程图,我们在线程中用handler通过sendEmptyMessage方法(或者sendMessage方法)发送消息存到MessageQueue中。而我们刚才的Looper类,是一个轮询器,它的作用就是一直从MsgQueue中获取消息,然后将消息发放到handler中,让他执行逻辑。

我们还是从looper.prepareMainLooper()开始看吧.


2018-04-12—(重点)源码角度分析Handler运行原理
prepareMainLooper方法

我们发现主类中执行的prepareMainLooper方法其实也是执行prepare方法,我们点进prepare方法。


2018-04-12—(重点)源码角度分析Handler运行原理
prepare方法

他在下面执行了一个ThreadLocal.set(new Looper(...))方法,我们暂时可以理解为我们在ThreadLocal中添加了一个 当前线程的looper对象。

我们继续看prepareMainLooper,他之后又执行了一个myLooper方法,我们点进去。

2018-04-12—(重点)源码角度分析Handler运行原理
myLooper方法

这个方法是将我们刚才存入的当前线程的looper对象取出来。

然后我们来看一下loop方法。


2018-04-12—(重点)源码角度分析Handler运行原理
loop方法

我在里面写了很清楚的备注,所以当我们执行了loop方法之后,他就会开始无效循环,一直在寻找当前线程的消息队列中是否有message,

2018-04-12—(重点)源码角度分析Handler运行原理
loop方法

我们接着看,在下面我们轮询时候消息队列有msg时候,我们会让msg的target对象执行dipatchMessage方法,我们看一看target属性是什么:


2018-04-12—(重点)源码角度分析Handler运行原理

我们发现他是个Handler类的属性,其实这个target就是发送msg的那个handler,为什么我们下面再说。

我们来接着看一下dispatchMessage方法,

2018-04-12—(重点)源码角度分析Handler运行原理
dispatchMessage

我们发现原来最后会回调到handleMessage方法中,这个方法我们应该很熟悉了,就是我们刚刚执行操作的方法。至于为什么target是我们发送msg的handler呢,我们现在来看一下:

我们从handler.sendMessage方法开始看吧:

2018-04-12—(重点)源码角度分析Handler运行原理
2018-04-12—(重点)源码角度分析Handler运行原理
2018-04-12—(重点)源码角度分析Handler运行原理
2018-04-12—(重点)源码角度分析Handler运行原理

他从sendMessage一直跳转了这么几个方法,在最后的恩queueMessage方法中,我们看到msg.target= this,将当前的handler设为msg的target,所以target存储的就是发送msg的handler。

我们来看一下最后跳转到消息队列的enqueueMessage方法:


2018-04-12—(重点)源码角度分析Handler运行原理
MessageQueue的enqueueMessage方法

这个方法其实就是一个链表插入,把我们的msg放入到消息队列中。



所以现在我们知道了,looper的loop方法一旦启动,就会一直获取当前线程的looper对象,然后一直轮询当前对象的消息队列,当找到消息就发放给msg的handler,让他执行handleMessage方法,这就是他的工作原理。

下面给大家配上一张逻辑图和源码结合的图:


2018-04-12—(重点)源码角度分析Handler运行原理