Android开发使用Message对象分发必备知识点详解

作者:长安皈故里 时间:2022-02-08 21:32:49 

Message的创建

消息Message一般不支持大家直接通过new的方式进行创建的,因为Message作为Android系统中使用频率非常高的一个对象,如果每次都泛滥的直接创建一个新的,对性能是有一定影响的,而通过对象池的方式进行复用 ,则是非常好的一种方式。

Message中就提供了这样的一个对象池(最大缓存消息数量为50):

Android开发使用Message对象分发必备知识点详解

通过链表的形式将一个个待复用的缓存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);
   }
}

源码一目了然,接下来我们一个个的进行分析:

  • 先检测Messagecallback是否为null,不为null就执行callbackrun方法调度执行,这个一般是如何传入的呢:

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()方法就并不再过多介绍了。

  • 然后检测HandermCallback是否为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

标签:Android,Message,对象
0
投稿

猜你喜欢

  • MyBatis Generator去掉生成的注解

    2022-08-29 22:01:03
  • 身份证号码验证算法深入研究和Java实现

    2023-05-18 08:17:42
  • Java中MultipartFile与File互转的实现

    2022-02-15 15:36:02
  • Android SDK在线更新镜像服务器大全

    2023-03-14 05:44:27
  • C#中的延时函数sleep

    2022-08-05 13:16:49
  • Java实现三子棋小游戏

    2022-09-12 01:27:20
  • Java 实现RSA非对称加密算法

    2021-08-19 16:27:51
  • java判断某个点是否在所画多边形/圆形内

    2022-09-30 23:50:45
  • C#使用RabbitMq队列(Sample,Work,Fanout,Direct等模式的简单使用)

    2023-12-06 10:45:34
  • 详解AndroidStudio3.0开发调试安卓NDK的C++代码

    2022-09-29 07:02:53
  • Java实现简单树结构

    2023-08-06 18:59:12
  • 基于mybatis plus实现数据源动态添加、删除、切换,自定义数据源的示例代码

    2021-08-20 20:07:46
  • Java获取用户IP属地模拟抖音详解

    2023-04-18 02:01:29
  • Android 官推 kotlin-first 的图片加载库——Coil的使用入门

    2022-07-06 00:53:34
  • 基于spring-boot和docker-java实现对docker容器的动态管理和监控功能[附完整源码下载]

    2022-02-04 00:41:18
  • Android基于TextView实现跑马灯效果

    2023-07-25 06:32:45
  • 详解Java中的流程控制

    2022-03-29 06:52:19
  • mybatis如何使用Criteria的and和or进行联合查询

    2023-02-23 00:44:13
  • Spring-data-redis操作redis cluster的示例代码

    2022-12-17 17:04:00
  • SpringBoot打jar包遇到的xml文件丢失的解决方案

    2023-04-11 23:39:06
  • asp之家 软件编程 m.aspxhome.com