【原创】源码角度分析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底层是基于 ...
随机推荐
- bzoj4031 [HEOI2015]小Z的房间
Description 你突然有了一个大房子,房子里面有一些房间.事实上,你的房子可以看做是一个包含n*m个格子的格状矩形,每个格子是一个房间或者是一个柱子.在一开始的时候,相邻的格子之间都有墙隔着. ...
- jQuery选择器与CSS选择器
1. 通过位置选择的几个操作: :first:默认情况下是相对整个页面来说的第一个,如:li:first表示整个页面的第一个li元素,而ul li:first表示整个页面的第一个li元素,并且是在ul ...
- scrapy(一)建立一个scrapy项目
本项目实现了获取stack overflow的问题,语言使用python,框架scrapy框架,选取mongoDB作为持久化数据库,redis做为数据缓存 项目源码可以参考我的github:https ...
- Nginx 反向代理&负载均衡
1.反向代理 当我们请求一个网站时,nginx会决定由哪台服务器提供服务,就是反向代理. nginx只做请求的转发,后台有多个tomcat服务器提供服务,nginx的功能就是把请求转发给后面的服务器, ...
- 让Cocos2dx中的TestCPP中的Box2dTest运行起来
一般而言,如果你导入TestCPP到VS2012中去后,会编译十几分钟才会出现窗口界面,这是包含Cocos2dx所有功能的一个demo,功能非常齐全强大,里面有两个关于Box2d的测试案例,一个是Bo ...
- js闭包绑定元素
闭包,官方对闭包的解释是:一个拥有许多变量和绑定了这些变量的环境的表达式(通常是一个函数),因而这些变量也是该表达式的一部分.闭包的特点: 1. 作为一个函数变量的一个引用,当函数返回时,其处于激活状 ...
- python3 time模块与datetime模块
time模块 在Python中,通常有这几种方式来表示时间:1)时间戳 2)格式化的时间字符串 3)元组(struct_time)共九个元素.由于Python的time模块实现主要调用C库,所以各个平 ...
- 基于ssh,shell,python,iptables,fabric,supervisor和模板文件的多服务器配置管理
前言:略 新服务器:NS 主服务器:OS 一:OS上新建模板目录例如 mkdir bright 用于导入一些不方便在远程修改的配置文件.redis.conf等,到需要配置的步骤时用远程cp命令覆 ...
- 配置nginx脚本开机自启动
vi /etc/init.d/nginx 插入以下内容,修改红色字体自己安装路径 #!/bin/bash## chkconfig: - 85 15# description: Nginx is a W ...
- windows上安装jdk
1.下载jdk安装包 jdk官网 http://www.oracle.com/technetwork/java/javase/downloads/index.html 2.安装jdk 傻瓜式下一步 ...