Android异步消息传递机制源码分析&&相关知识常被问的面试题

1、Android异步消息传递机制有以下两个方式:(异步消息传递来解决线程通信问题)

handler 和 AsyncTask

2、handler官方解释的用途:

1)、定时任务:通过handler.postDelay(Runnable r, time)来在指定时间执行msg.

2)、线程间通信:在执行较为耗时操作的时候,在子线程中执行耗时任务,然后handler(主线程的)把执行的结果通过sendmessage的方式发送给UI线程去执行用于更新UI.

3、handler源码分析

一、在ActivityThread.main(应用程序的入口函数,主线程中执行)

看到第5401行代码功能是生成了一个主线程的Looper对象。

100->90行生成Lopper并扔到ThreadLocal中。

105标记生成的Lopper为主线程的Looper。

5417功能是,新生成的Lpooer不断去loop消息队列,消息队列后面介绍。

一个线程只能有一个Lopper, 主线程默认在启动时候已经由系统创建完成,这也就代表,我们自己写的线程要使用消息传递机制,需要自己生成looper并调用looper.loop去循环消息队列。

看下looper.loop循环的是什么?

127看到一个looper有一个MessageQueue,134开始是不断的从messageQueue中取出message交给148msg.target.dispatchMessage去处理。

135:面试点

二、看下消息传递的源码

handler属于它建立的线程。也就是说handler如果在主线程中创建,那么他处理消息就是在主线程中处理,如果实在子线程中建立,那么消息的处理就在子线程中进行。

1、我们都知道通过handler发送消息有以下几种方式:

handler.sendMessage(msg);

hadler.sendMessageDelayed(msg, time);

sendEmptyMessage(int what);

public final boolean sendEmptyMessageDelayed(int what, long delayMillis)

其中,sendEmptyMessage的实现

只是把传进去的what写在了message里

几个函数最后都变成调用public boolean sendMessageAtTime(Message msg, long uptimeMillis)

600行把msg放到了Looper的messageQueue中。

接下来看enqueueuMessage中做了什么:

551开始,根据msg中携带的时间信息,找到当前时间要执行的msg,放在messageQueue队首。

那消息在消息队列中是怎么样被执行的呢,这就要看下loop函数

主要看下148行,是把msg交给msg.target的dispatchMessage函数来执行的,那msg.target又是哪个对象呢?

handler的enqueueMessage中看到msg.target就是handler.

所以可以看到msg最终是在handler的dispatchMessage中执行的。

看到这个函数中首先哦按段msg.callback为空的话就会调用handler的handleMessage.

至此handler处理消息的整个源码过程分析完毕。

三、除了发送消息之外,还有一下几种方式可以在子线程中发送UI跟新操作。

Handler.post

View.post

Activity.runonUithread

1、Handler.post

还是调用的sendMessageDelayed,看下getPostMeassage源码

看到msg的callback被设置成了Runnable。再回头看下上面dispatchMessage的源码,这时候msg.callback不会空,回去执行handlecallback

直接执行Runnable中的run方法。

总结一下,也就是说使用handler的post方法相当于是把带callback的msg入队列了,然后在去除msg时候,直接调用的callback的run方法。

注意点:google建议我们在写message的时候,最好使用Message.obtain()或者Handler.obtainMessage()来获取当前looper的message,而不是自己去new Message,这样做的好处是Message.obtain()会从消息池中获取一个Message对象,如果消息池中是空的,才会使用构造方法实例化一个新Message,这样有利于消息资源的利用,msg也可以更好的被回收池回收。

2、View.post

就是调用的handler的post方法,所以同上一个流程。

3、Activity.runonUithread

看实现功能是,查看当前线程是否是主线程,如果是直接执行run方法,不是的话,调用handler的post方法,注释解析的也是再清晰不过了。

补充:看下message的几个成员变量

在我们传递message的时候可以设置如上变量值(如下4条参考http://blog.csdn.net/ahuier/article/details/17012005

1) arg1 和 arg2 都是Message自带的用来传递一些轻量级存储int类型的数据,比如进度条的数据等。通过这个数据是通过Bundle的方式来转载的,读者可以自己查阅源代码研究。

2) obj 是Message自带的Object类型对象,用来传递一些对象。兼容性最高避免对齐进行类型转换等。

3) replyTo 是作为线程通信的时候使用.

4) what 用户自定义的消息码让接受者识别消息种类,int类型。

四、常被问的相关面试题。

1、Looper.loop在message为空的时候会一直去loop吗?

解答:https://www.zhihu.com/question/34652589

(连接中内容,强烈建议看下知乎完整解答,写的很好)这里就涉及到Linux pipe/epoll机制,简单说就是在主线程的MessageQueue没有消息时,便阻塞在loop的queue.next()中的nativePollOnce()方法里,此时主线程会释放CPU资源进入休眠状态,直到下个消息到达或者有事务发生,通过往pipe管道写端写入数据来唤醒主线程工作。这里采用的epoll机制,是一种IO多路复用机制,可以同时监控多个描述符,当某个描述符就绪(读或写就绪),则立刻通知相应程序进行读或写操作,本质同步I/O,即读写是阻塞的。 所以说,主线程大多数时候都是处于休眠状态,并不会消耗大量CPU资源。

2、sendMessage和post什么区别呢?

我想看完上面的分析,自然就会通了

3、msg.target是谁?

同2

时间: 2024-10-10 20:01:24

Android异步消息传递机制源码分析&&相关知识常被问的面试题的相关文章

拨云见日---android异步消息机制源码分析

做过windows GUI的同学应该清楚,一般的GUI操作都是基于消息机制的,应用程序维护一个消息队列,开发人员编写对应事件的回调函数就能实现我们想要的操作 其实android系统也和windows GUI一样,也是基于消息机制,今天让我们通过源码来揭开android消息机制的神秘面纱 谈起异步消息,就不能不提及Handler,在安卓中,由于主线程中不能做耗时操作,所以耗时操作必须让子线程执行,而且只能在主线程(即UI线程)中执行UI更新操作,通过Handler发送异步消息,我们就能更新UI,一

Android事件分发机制源码分析

小小感慨一下,做android有一段时间了,一直以来都是习惯整理笔记存到有道笔记上,没有写博客的习惯.以后逐步分类整理出来,也算"复习"一遍了 - _ - . android的事件分发相关的方法有三个: 1.public booleandispatchTouchEvent(MotionEvent ev) 2.public boolean onInterceptTouchEvent(MotionEvent ev) 3.public booleanonTouchEvent(MotionEv

Android 中View的绘制机制源码分析 三

到目前为止,measure过程已经讲解完了,今天开始我们就来学习layout过程,不过在学习layout过程之前,大家有没有发现我换了编辑器,哈哈,终于下定决心从Html编辑器切换为markdown编辑器,这里之所以使用"下定决心"这个词,是因为毕竟Html编辑器使用好几年了,很多习惯都已经养成了,要改变多年的习惯确实不易,相信这也是还有很多人坚持使用Html编辑器的原因.这也反应了一个现象,当人对某一事物非常熟悉时,一旦出现了新的事物想取代老的事物时,人们都有一种抵触的情绪,做技术的

Android -- 消息处理机制源码分析(Looper,Handler,Message)

android的消息处理有三个核心类:Looper,Handler和Message.其实还有一个Message Queue(消息队列),但是MQ被封装到Looper里面了,我们不会直接与MQ打交道,因此我没将其作为核心类.下面一一介绍: Looper Looper的字面意思是“循环者”,它被设计用来使一个普通线程变成Looper线程.所谓Looper线程就是循环工作的线程.在程序开发中(尤其是GUI开发中),我们经常会需要一个线程不断循环,一旦有新任务则执行,执行完继续等待下一个任务,这就是Lo

Android 中View的绘制机制源码分析 二

尊重原创:http://blog.csdn.net/yuanzeyao/article/details/46842891 本篇文章接着上篇文章的内容来继续讨论View的绘制机制,上篇文章中我们主要讲解了View的measure过程,今天我们就来学习ViewGroup的measure过程,由于ViewGroup只是一个抽象类,所以我们需要以一个具体的布局来分析measure过程,正如我上篇文章说的,我打算使用LinearLayout为例讲解measure过程,如果你还没有读过上篇文章,那么建议你先

Android 上千实例源码分析以及开源分析

Android 上千实例源码分析以及开源分析(百度云分享) 要下载的直接翻到最后吧,项目实例有点多. 首先 介绍几本书籍(下载包中)吧. 01_Android系统概述 02_Android系统的开发综述 03_Android的Linux内核与驱动程序 04_Android的底层库和程序 05_Android的JAVA虚拟机和JAVA环境 06_Android的GUI系统 07_Android的Audio系统 08_Android的Video 输入输出系统 09_Android的多媒体系统 10_

深度理解Android InstantRun原理以及源码分析

深度理解Android InstantRun原理以及源码分析 @Author 莫川 Instant Run官方介绍 简单介绍一下Instant Run,它是Android Studio2.0以后新增的一个运行机制,能够显著减少你第二次及以后的构建和部署时间.简单通俗的解释就是,当你在Android Studio中改了你的代码,Instant Run可以很快的让你看到你修改的效果.而在没有Instant Run之前,你的一个小小的修改,都肯能需要几十秒甚至更长的等待才能看到修改后的效果. 传统的代

Spark资源调度机制源码分析--基于spreadOutApps及非spreadOutApps两种资源调度算法

Spark资源调度机制源码分析--基于spreadOutApps及非spreadOutApps两种资源调度算法 1.spreadOutApp尽量平均分配到每个executor上: 2.非spreadOutApp尽量在使用单个executor的资源. 源码分析 org.apache.spark.deploy.master.Master 1.首先判断,master状态不是ALIVE的话,直接返回2.调度driver3. Application的调度机制(核心之核心,重中之重) 源码如下: 1 /*

Java NIO——Selector机制源码分析---转

一直不明白pipe是如何唤醒selector的,所以又去看了jdk的源码(openjdk下载),整理了如下: 以Java nio自带demo : OperationServer.java   OperationClient.java(见附件) 其中server端的核心代码: public void initSelector() { try { selector = SelectorProvider.provider().openSelector(); this.serverChannel1 =