Android开发使用Message对象分发必备知识点详解
作者:长安皈故里 时间:2022-02-08 21:32:49
Message的创建
消息Message
一般不支持大家直接通过new
的方式进行创建的,因为Message
作为Android系统中使用频率非常高的一个对象,如果每次都泛滥的直接创建一个新的,对性能是有一定影响的,而通过对象池的方式进行复用 ,则是非常好的一种方式。
Message
中就提供了这样的一个对象池(最大缓存消息数量为50):
通过链表的形式将一个个待复用的缓存Message
连接起来。并且提供了obtain()
方法负责从对象池中获取一个Message
:
public static Message obtain() {
synchronized (sPoolSync) {
if (sPool != null) {
Message m = sPool;
sPool = m.next;
m.next = null;
m.flags = 0; // clear in-use flag
sPoolSize--;
return m;
}
}
return new Message();
}
当消息调度完毕时,会通过recycleUnchecked()
方法进行回收并放入到对象池:
void recycleUnchecked() {
flags = FLAG_IN_USE;
what = 0;
arg1 = 0;
arg2 = 0;
obj = null;
...
synchronized (sPoolSync) {
if (sPoolSize < MAX_POOL_SIZE) {
next = sPool;
sPool = this;
sPoolSize++;
}
}
}
重置要回收的Message
的各个成员属性,然后添加到对象池sPool
。
消息分发执行的三种方式
消息调度分发最终是在Looper.loopOnce()
中执行,我们看下源码:
private static boolean loopOnce(final Looper me,
final long ident, final int thresholdOverride) {
Message msg = me.mQueue.next(); // might block
if (msg == null) {
return false;
}
try {
msg.target.dispatchMessage(msg);
}
msg.recycleUnchecked();
return true;
}
核心就是msg.target.dispatchMessage()
,我们看下具体的方法逻辑:
public void dispatchMessage(@NonNull Message msg) {
if (msg.callback != null) {
handleCallback(msg);
} else {
if (mCallback != null) {
if (mCallback.handleMessage(msg)) {
return;
}
}
handleMessage(msg);
}
}
源码一目了然,接下来我们一个个的进行分析:
先检测
Message
的callback
是否为null,不为null就执行callback
的run
方法调度执行,这个一般是如何传入的呢:
public final boolean post(@NonNull Runnable r) {
return sendMessageDelayed(getPostMessage(r), 0);
}
private static Message getPostMessage(Runnable r) {
Message m = Message.obtain();
m.callback = r;
return m;
}
很熟悉的post()
方法就并不再过多介绍了。
然后检测
Hander
的mCallback
是否为null,不为null就执行mCallback.handleMessage()
,这个是什么时候传入的呢:
public Handler(@NonNull Looper looper, @Nullable Callback callback, boolean async) {
mLooper = looper;
mQueue = looper.mQueue;
mCallback = callback;
mAsynchronous = async;
}
通过Handler
构造方法时作为构造参数传入,可以选择性使用。
上面都不满足,就调用
Handler
自身的handleMessage()
方法调度执行,这也是我们常用的消息执行的一种方式,一般都是创建Handler
对象时重写这个该方法:
fun test333() {
val handler = object : Handler(Looper.getMainLooper()) {
override fun handleMessage(msg: Message) {
super.handleMessage(msg)
}
}
}
以上三种消息分发方式第一种和第三种使用的频率比较高,第二种这种方式可以作为一种hook的手段拦截某些消息的原本调度逻辑,实现功能增强。
比如Activity、Service
等组件的调度是通过ApplicationThread
通过Handler
分发到主线程进行调度执行,如果你想监听其生命周期,就可以通过上面的第二种方式结合反射给负责分发的Handler
注入一个mCallback
属性值。
来源:https://juejin.cn/post/7151929832698281991
![](/images/zang.png)
![](/images/jiucuo.png)
猜你喜欢
MyBatis Generator去掉生成的注解
身份证号码验证算法深入研究和Java实现
Java中MultipartFile与File互转的实现
![](https://img.aspxhome.com/file/2023/9/83079_0s.png)
Android SDK在线更新镜像服务器大全
![](https://img.aspxhome.com/file/2023/8/103878_0s.jpg)
C#中的延时函数sleep
Java实现三子棋小游戏
Java 实现RSA非对称加密算法
java判断某个点是否在所画多边形/圆形内
C#使用RabbitMq队列(Sample,Work,Fanout,Direct等模式的简单使用)
![](https://img.aspxhome.com/file/2023/2/83302_0s.png)
详解AndroidStudio3.0开发调试安卓NDK的C++代码
![](https://img.aspxhome.com/file/2023/6/97866_0s.png)
Java实现简单树结构
基于mybatis plus实现数据源动态添加、删除、切换,自定义数据源的示例代码
Java获取用户IP属地模拟抖音详解
![](https://img.aspxhome.com/file/2023/7/81897_0s.jpg)
Android 官推 kotlin-first 的图片加载库——Coil的使用入门
![](https://img.aspxhome.com/file/2023/7/89527_0s.png)
基于spring-boot和docker-java实现对docker容器的动态管理和监控功能[附完整源码下载]
![](https://img.aspxhome.com/file/2023/6/82446_0s.png)
Android基于TextView实现跑马灯效果
详解Java中的流程控制
mybatis如何使用Criteria的and和or进行联合查询
Spring-data-redis操作redis cluster的示例代码
SpringBoot打jar包遇到的xml文件丢失的解决方案
![](https://img.aspxhome.com/file/2023/4/66174_0s.png)