「Android」消息驱动Looper和Handler类分析
Android系统中的消息驱动工作原理:
1、有一个消息队列,可以往这个消息队列中投递消息;
2、有一个消息循环,不断的从消息队列中取得消息,然后处理。
工作流程:
1、事件源将待处理的消息加入到消息队列中,一般是加至队列尾部(优先级高的消息可以加至队列头),事件源提交的消息可以是按键、触摸等物理事件产生的消息,也可以是系统或应用程序发出的消息;
2、处理线程不断的从消息队列头中取出消息并处理。
在Android系统中,这些工作主要由Looper和Handler实现:
Looper类,用于封装消息循环,并且有一个消息队列;
Handler类,封装消息投递、消息处理等接口。
一、Looper类分析
Looper用于封装了android线程中的消息循环,默认情况下一个线程是不存在消息循环(message loop)的,需要调用Looper.prepare()来给线程创建一个消息循环,调用Looper.loop()来使消息循环起作用,使用Looper.prepare()和Looper.loop()创建了消息队列就可以让消息处理在该线程中完成。
class LooperThread extends Thread
{
public void run()
{
Looper.prepare();
//代码1....
Looper.loop();
//代码2....
}
}
(1)在 Looper.prepare()中,定义一个ThreadLocal对象sT,并构造一个Looper对象设置到调用线程的局部变量sT中。
ThreadLocal是java中的线程局部变量类(Thread Local Variable),该类有两个关键函数:
- set:设置调用线程的局部变量
- get:获取调用线程的局部变量
因此,prepareh函数会在调用线程的局部变量中设置一个Looper对象。这个调用线程就是LooperThread的run线程。
private Looper(){
//构造一个消息队列
mQueue = new MessageQueue();
//得到当前线程的Thread对象
mThread = Thread.currentThread();
}
prepare函数中设置了一个Looper对象,对象保存在这个调用线程的局部变量中。而Looper 对象内部封装了一个消息队列。
(2)Looper循环
> 在Loop()方法中,首先通过ThreadLocal的get方法获取创建的Looper对象;
> 之后取出这个Looper的消息队列 MessageQueue queue = looper.mQueue;
> while循环中,处理消息:
while(true){
Message msg = queue.next();
// 处理i消息,Message对象中有一个target(Handler类型)
// 如果target为空,则退出消息循环
if (msg != null ) {
if (msg.target == null) { return; }
// 调用该消息的Handler,交给他的dispatchMessage函数处理。
msg.target.dispatchMessage(msg);
msg.recyle();
}
}
因此,分析prepare和loop函数后,Looper的作用有:
1、封装了一个消息队列;
2、prepare函数把这个Looper和调用prepare的线程(最终的处理线程)关联在i一起;
3、处理线程调用loop函数,处理来自消息队列的消息
当事件源向Looper发送消息时,消息加到Looper的消息队列。该消息将由和Looper绑定的处理线程来处理。
注:Looper、Message、Handler的关系:
1、Looper中有一个Message队列,存储的时一个个待处理的message;
2、Message中有一个Handler,用来处理Message。
二、Handler类分析
Handler中包括的成员:
Hadler.java:
final messageQueue mQueue; //Handler中也有一个消息队列
final Looper mLooper;
final Callback mCallback; //i回调用的类
在Handler的构造函数中,Handler中的消息队列最终会指向Looper的消息队列。
(1)插入消息到Looper消息队列
Handler提供了一系列帮助完成创建消息和插入消息队列的函数。
例如Handler.java中sendMessage发送一个消息,将消息添加到消息队列末尾:
public final boolean sendMessage(Message msg) {
return sendMessageDelayed(msg, );
}
public final boolean senMessageDelayed(Message msg, long delayMilis){
......
return senMessage(msg, SystemClock.uptimeMillis() + delayMillis); // 当前时间
}
public boolean sendMessageTime(Message msg, long uptimeMillis) {
boolean sent = false;
MessageQueue queue = mQueue;
if (queue != null) {
msg.target = this; // 将targt设置为自己,然后加入到消息队列
sent = queue.enqueueMessage(msg, uptimeMillis);
}
return sentl
}
(2)Handler的消息处理
在loop方法中,如果获得消息后会调用target的dispatchMessage函数,然后将这个消息派发给Handler处理。
dispatchMessage定义了一套消息处理的优先级机制:
> Messagu如果自带callback处理,则交给callback处理;
> Handler如果设置了全局的mCallback,则交给mCallback处理;
> 如果上述都没有,消息则会被交给Handler子类实现的handleMessage 来处理。(这需要从Handler派生并重载handleMessage函数)
「Android」消息驱动Looper和Handler类分析的更多相关文章
- Android的消息循环机制 Looper Handler类分析
Android的消息循环机制 Looper Handler类分析 Looper类说明 Looper 类用来为一个线程跑一个消息循环. 线程在默认情况下是没有消息循环与之关联的,Thread类在ru ...
- Android中消息系统模型和Handler Looper
http://www.cnblogs.com/bastard/archive/2012/06/08/2541944.html Android中消息系统模型和Handler Looper 作为Andro ...
- Android的消息机制: Message/MessageQueue/Handler/Looper
概览 * Message:消息.消息里面可包含简单数据.Object和Bundle,还可以包含一个Runnable(实际上可看做回调). * MessageQueue:消息队列,供Looper线程 ...
- Android之MessageQueue、Looper、Handler与消息循环
在android的activity中有各种各样的事件,而这些事件最终是转换为消息来处理的.android中的消息系统涉及到: * 消息发送 * 消息队列 * 消息循环 * 消息分发 * 消息 ...
- 「Android」SurfaceFlinger分析
本篇针对surfaceFlinger模块进行分析,目录如下: 1.SurfaceFlinger功能 1.1.BufferQueue原理(native/libs/gui模块) 1.2 layer显示 ...
- Android应用程序消息处理机制(Looper、Handler)分析
文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/6817933 Android应用程序是通过消息来 ...
- 「Android」 Surface分析
本篇针对Surface模块进行分析,从Java层的Activity创建开始,到ViewRoot.WindowsManagerService,再到JNI层和Native层. 首先推荐一个Android源 ...
- 「Android」adb调试源码(针对dumpsys SurfceFlinger、trace.txt获取)
首先对ADB作简单的阐述,接下来对adb shell dumpsys SurfaceFlinger服务的dump信息的查看.以及ANR问题如何获取trace文件并简单分析. -×*********** ...
- 「Android」系统架构概述
目录: 1.Android系统架构 2.Android类库 3.四大组件 --------------------------------------------------------------- ...
随机推荐
- Kubernetes 笔记 02 demo 初体验
本文首发于我的公众号 Linux云计算网络(id: cloud_dev),专注于干货分享,号内有 10T 书籍和视频资源,后台回复「1024」即可领取,欢迎大家关注,二维码文末可以扫. 从前面的文章我 ...
- 根据sockfd获取TCP连接本地地址以及对端地址
摘抄自muduo代码实现. 获取本地地址(LocalAddr): struct sockaddr_in6 sockets::getLocalAddr(int sockfd) { struct sock ...
- 【code block】局部代码块+构造代码块+静态代码块
1.局部代码块 位置:位于类的方法中 表示方法:{} 作用:控制变量的生命周期,减少内存消耗 demo: public class LocalCode { public static void mai ...
- SQL笔记之SELECT语句
SELECT语句的完整语法: SELECT[ALL/DISTINCT]目标列表达式 [AS 别名],··· --ALL不去掉重复 DISTINCT去掉重复 FROM 表名或视图名 或者(SELECT ...
- 使用C# (.NET Core) 实现迭代器设计模式 (Iterator Pattern)
本文的概念来自深入浅出设计模式一书 项目需求 有两个饭店合并了, 它们各自有自己的菜单. 饭店合并之后要保留这两份菜单. 这两个菜单是这样的: 菜单项MenuItem的代码是这样的: 最初我们是这样设 ...
- 扒一扒.net、.net framework、mono和Unity
zhaichao 标签: .net.net frameworkc#monounity 2017-04-23 14:39 425人阅读 评论(0) 收藏 举报 版权声明:本文为博主原创文章,未经博主允许 ...
- 【WebAPI】从零开始学会使用.NET Core WebAPI
介绍 以后会慢慢总结在项目使用中或者学习到的webAPI相关的知识,在这里做记录. 我会从最开始的如何创建WebAPI项目到项目的后续知识一点一点的开始讲述记录. 通过简单有效的方式,让我们能够快速的 ...
- JVM学习(一)、垃圾收集器简介
一.垃圾收集算法 (1)标记-清除算法:最基础的收集算法“标记--清除”(Mark-sweep)算法,算法分为“标记”和“清除”两个阶段:首先标记出所有需要回收的对象,在标记完成后统一回收所有被标记的 ...
- leetcode — longest-valid-parentheses
import java.util.Stack; /** * Source : https://oj.leetcode.com/problems/longest-valid-parentheses/ * ...
- Git命令速查
Alias 下面的只是例子,想改成什么跟随自己的意愿即可. git config --global alias.st status //status 缩写成 st git config --globa ...