【原创】源码角度分析Android的消息机制系列(一)——Android消息机制概述
ι 版权声明:本文为博主原创文章,未经博主允许不得转载。
1.为什么需要Android的消息机制
因为Android系统不允许在子线程中去访问UI,即Android系统不允许在子线程中更新UI。
为什么不允许在子线程中更新UI呢?因为Android的控件不是线程安全的。既然是非线程安全的,那么若在多个子线程中并发访问,UI控制可能会处于一种不可预期的状态。有的读者可能会说,为什么不对UI控件加锁呢?加锁会降低UI访问的效率,因为加锁之后,若想要运行这段synchronized的代码,线程要先拿到执行这段代码的权限,Java里面也即拿到某个同步对象的锁,但是一个对象只有一把锁,若此时这个同步对象的锁被其他线程拿走了,那这个线程就只能先在等待队列中等待了,而且当持有对象锁的那个线程执行完,释放对象锁后,不一定会唤醒该线程,那么该线程等待的时间也是未知的。这样一来,就会导致访问UI的效率很低。
Android的消息机制就是为了解决在子线程中访问UI这一问题而诞生的。
2.何为Android的消息机制
Android的消息机制主要是指Handler的运行机制。
说到Handler,必然要提及另外三个概念,即Message、MessageQueue、Looper。而这四者的关系也常常是Android开发者在求职面试中经常被问及的一个问题。下面我们先简单说明一下四者的作用以及关系,关于其实现原理,我们会在后续博文中详细介绍。
Message:消息,当定义一个Message时,包含必要的描述和属性数据。常用属性:arg1、arg2、what、obj、target等,其中arg1和arg2可以存放整型数据,what可以用来标识一条Message,obj可以存放Object类型的任意对象,target就是处理一个Message的Handler。一般情况下,Message不需要new出来,可以调用handler的obtainMessage()方法获取一个Message。
MessageQueue:消息队列,其内部维护了一组消息,以队列的形式对外提供了插入和读取的操作。虽然名字为消息队列,但其内部并非真正的队列,而是采用了单链表的数据结构来存储了消息列表。
Looper:循环,在此即消息循环,Looper并非简单的循环,而是可以处理消息的。Looper会无限循环的去查找是否有新的消息,有的话,则处理消息,否则,就一直等待着。Looper中有一个比较特殊的概念,即ThreadLocal,其并非线程,而是在每个线程中存储数据用的。线程默认是没有Looper的,但是当ActivityThread(主线程,即UI线程)被创建时,就会初始化Looper。
Handler:可以将一个任务切换到某个指定的线程中去执行,一般可以用其更新UI。Handler创建的时候会采用当前线程的Looper来构造消息循环系统,Handler通过ThreadLocal可以获取到当前线程的Looper,又因为ActivityThread在被创建的时候就初始化Looper了,所以在主线程中默认是可以使用Handler的。
Handler的运行需要MessageQueue和Looper的支撑,当Hanler创建完毕后,其内部的Looper和MessageQueue就可以和Handler一起协同工作了。Handlr可以通过两种方式发送消息:①在不同线程之间,可以通过send方式发送;②在未来某个时间执行任务,可以用post方式,post方法将一个Runnable投递到Handler的内部Looper中去处理,其实post方法最终也是通过send方法来完成的。
Handler的send方法被调用后,MessageQueue的enqueueMessage会将这个Message放到消息队列中,Looper检测到有新的消息到来了,就会去处理这个消息,最终消息中的Runnable或Handller的handlerMessage方法会被调用,消息被处理。而Looper是运行在创建Handler的线程中的,这样一来,Handler中的业务逻辑就被切换到创建Handler所在的线程中去执行了。我们就可以在主线程中创建Handler,然后将更新UI的重任放在Handler的业务逻辑中了。当需要在子线程中更新UI时,我们就可以只和Handler进行交互了。
【原创】源码角度分析Android的消息机制系列(一)——Android消息机制概述的更多相关文章
- 源码角度分析-newFixedThreadPool线程池导致的内存飙升问题
前言 使用无界队列的线程池会导致内存飙升吗?面试官经常会问这个问题,本文将基于源码,去分析newFixedThreadPool线程池导致的内存飙升问题,希望能加深大家的理解. (想自学习编程的小伙伴请 ...
- 【原创】源码角度分析Android的消息机制系列(五)——Looper的工作原理
ι 版权声明:本文为博主原创文章,未经博主允许不得转载. Looper在Android的消息机制中就是用来进行消息循环的.它会不停地循环,去MessageQueue中查看是否有新消息,如果有消息就立刻 ...
- 【原创】源码角度分析Android的消息机制系列(三)——ThreadLocal的工作原理
ι 版权声明:本文为博主原创文章,未经博主允许不得转载. 先看Android源码(API24)中对ThreadLocal的定义: public class ThreadLocal<T> 即 ...
- 【原创】源码角度分析Android的消息机制系列(六)——Handler的工作原理
ι 版权声明:本文为博主原创文章,未经博主允许不得转载. 先看Handler的定义: /** * A Handler allows you to send and process {@link Mes ...
- 【原创】源码角度分析Android的消息机制系列(四)——MessageQueue的工作原理
ι 版权声明:本文为博主原创文章,未经博主允许不得转载. MessageQueue,主要包含2个操作:插入和读取.读取操作会伴随着删除操作,插入和读取对应的方法分别为enqueueMessage和ne ...
- Android的Message Pool是什么——源码角度分析
原文地址: http://blog.csdn.net/xplee0576/article/details/46875555 Android中,我们在线程之间通信传递通常采用Android的消息机制,而 ...
- 从源码角度分析 MyBatis 工作原理
一.MyBatis 完整示例 这里,我将以一个入门级的示例来演示 MyBatis 是如何工作的. 注:本文后面章节中的原理.源码部分也将基于这个示例来进行讲解.完整示例源码地址 1.1. 数据库准备 ...
- Adroid学习之 从源码角度分析-禁止使用回退按钮方案
有时候,不能让用户进行回退操作,如何处理? 查看返回键触发了哪些方法.在打开程序后把这个方法禁止了. 问题:程序在后台驻留,这样就会出现,其他时候也不能使用回退按钮.如何处理,在onpase()时方法 ...
- 从源码角度分析 Kotlin by lazy 的实现
by lazy 的作用 延迟属性(lazy properties) 是 Kotlin 标准库中的标准委托之一,可以通过 by lazy 来实现. 其中,lazy() 是一个函数,可以接受一个 Lamb ...
- Java面试题 从源码角度分析HashSet实现原理?
面试官:请问HashSet有哪些特点? 应聘者:HashSet实现自set接口,set集合中元素无序且不能重复: 面试官:那么HashSet 如何保证元素不重复? 应聘者:因为HashSet底层是基于 ...
随机推荐
- codeforces 757F Team Rocket Rises Again
链接:http://codeforces.com/problemset/problem/757/F 正解:灭绝树. mdzz倍增lca的根节点深度必须是1..我因为这个错误调了好久. 我们考虑先求最短 ...
- ICDM评选:数据挖掘十大经典算法
原文地址:http://blog.csdn.net/aladdina/article/details/4141177 国际权威的学术组织the IEEE International Conferenc ...
- hdu2819二分图匹配
Given an N*N matrix with each entry equal to 0 or 1. You can swap any two rows or any two columns. C ...
- MapControl Application 添加自定义的工具条
现在想用二次开发做一些东西,然后需要自定义的工具条,但是如何向MapControl Application 添加自定义的工具条呢,经过多次试验后,终于找到了相应的方法(左图是添加自定义的工具条之前,右 ...
- PHP 无限级分类(递归)
网上有很多,这是我自己做测试用的$arr = array( array('id'=>1,'name'=>'电脑','pid'=>0), array('id'=>2,'name' ...
- Python之函数知识
Python函数分类 a,内置函数 b,自定义函数 c,导入函数 一个函数就相当于一个功能块,比如获取数据库,更新数据库,函数其实就是代码的分块,调用函数来执行代码块 一块就代表一个功能 内置函数有以 ...
- hdu2089 不要62 我的第一个数位DP
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2089 数位DP的入门题,我是根据kuangbin的博客写出来的 思路: dp[i][0],表示长度为i ...
- 蚂蚁金服新一代数据可视化引擎 G2
新公司已经呆了一个多月,目前着手一个数据可视化的项目,数据可视化肯定要用到图形库如D3.Highcharts.ECharts.Chart等,经决定我的这个项目用阿里旗下蚂蚁金服所开发的G2图表库. 官 ...
- [大数据]-Elasticsearch5.3.1 IK分词,同义词/联想搜索设置
--题外话:最近发现了一些问题,一些高搜索量的东西相当一部分没有价值.发现大部分是一些问题的错误日志.而我是个比较爱贴图的.搜索引擎的检索会将我们的博文文本分词.所以图片内容一般是检索不到的,也就是说 ...
- Day4-内置函数--未完待续,慢慢写
内置函数:https://docs.python.org/3/library/functions.html?highlight=built#ascii 未完待续....