Android消息机制 知识点总结
1. ThreadLocal总结
ThreadLocal是一个线程内部的数据存储类,通过它可以在指定的线程中存储数据;不同线程对应不同的值。
用法简介
//创建
ThreadLocal<String> mStringThreadLocal = new ThreadLocal<>();
//set方法
mStringThreadLocal.set("yasinyao.com");
//get方法
mStringThreadLocal.get();
总结:实际上ThreadLocal的值是放入了当前线程的一个ThreadLocalMap实例中,所以只能在本线程中访问,其他线程无法访问。
注意:Android的ThreadLocal与Java实现略有不同,但是原理是一致的。
2. MessageQueue总结
Android消息队列在Android中是指MessageQueue,MessageQueue主要包括插入(enqueueMessage方法)和读取(next方法)
消息队列是一个单链表来维护消息列表,便于删除和添加数据
3.Looper总结
Looper在消息机制中扮演消息循环的角色,他会不同的从MessageQueue中查看是否有新消息,有旧处理,否则就一直阻塞在那里
Looper通过Looper.prepare()方法来初始化Looper,通过Looper.loop()方法开始处理消息
在Activity的中因为会默认初始化mainLooper 执行过loop()方法所以可以直接使用Looper
- Looper提供了quit和quitSafely来退出Looper
- quit 是直接退出Looper
- quitSafely 是添加推出的标记,等待消息都处理完成后再安全的推出
4.Handle总结
handle的工作主要是消息的发送和处理过程
消息的发送
- send方法
- post方法
- post的一系列方法最终都转化成send的一系列方法来实现的
消息的处理
- 核心方法dispatchMessage(Message msg)
- 钩子方法handleMessage(Message msg)
5.主线程的消息循环
Android的主线程就是ActivityThread,主线程的入口为main()方法。在main()中系统会通过Looper.perpareMainLooper()来创建主线程的Looper以及MessageQueue,并通过Looper.loop()来开启主线程消息循环
- Android中为什么主线程不会因为Looper.loop()的循环而卡死?
- Looper上的阻塞,前提是没有输入事件,MsgQ为空,Looper空闲状态,线程进入阻塞,释放CPU执行权,等待唤醒。
- UI耗时导致卡死,前提是要有输入事件,MsgQ不为空,Looper正常轮询,线程并没有阻塞,但是该事件执行时间过长(5秒?),而且与此期间其他的事件(按键按下,屏幕点击..)都没办法处理(卡死),然后就ANR异常了