Matrix源码分析(五)之 IdleHandlerLagTracer工作原理

从字面意思这个 IdleHandlerLagTracer 应该是监听 IdleHandler 的执行情况,不得不感叹Matrix功能做的是非常的细致,由于 IdleHandlerLagTracer 的代码量比较少,而且启动的重要成员的工作比较普通,我们先来介绍一下他的工作原理,再来看代码

IdleHandlerLagTracer 工作原理

1:IdleHandler 是Looper 的闲时机制,IdleHandlerLagTracer 会在开始工作后会启动一个 HandlerThread

2:使用反射将 MessageQueue 中承载着所有 IdleHandler 的 mIdleHandlers 这个 ArrayList 替换成自己的 新写的 MyArrayList ,这个这 MyArrayList 的add 和 remove 方法, 在调用 add 方法时,使用自己的 MyIdleHandler 作为中间层来传递事件,

3:在传递开始时使用绑定 HandlerThread 的 Handler 发送一个定时2s的延迟消息,如果2s后这个消息没有被移除,那么就会触发他的上报机制

在分析他的工作原理时将 IdleHandlerLagTracer 的整个工作流程分为了3部分,那么在分析源码的过程中,也使用这个流程

scss 复制代码
@Override
public void onAlive() {
    super.onAlive();
    if (traceConfig.isIdleHandlerTraceEnable()) {
        // 创建承载定时任务的 HandlerThread 
        idleHandlerLagHandlerThread = new HandlerThread("IdleHandlerLagThread");
        // 创建上报的任务,也就是定时2s后执行的任务
        idleHandlerLagRunnable = new IdleHandlerLagRunable();
        //反射 MessageQueue 中装在 IdleHandler 的 ArrayList
        detectIdleHandler();
    }
}

在开始先创建了一个 HandlerThread , 同时将上报的任务也创建了出来,最后在进行反射,来看看反射的情况

ini 复制代码
private static void detectIdleHandler() {
    try {
        if (android.os.Build.VERSION.SDK_INT < android.os.Build.VERSION_CODES.M) {
            return;
        }
        MessageQueue mainQueue = Looper.getMainLooper().getQueue();
        Field field = MessageQueue.class.getDeclaredField("mIdleHandlers");
        field.setAccessible(true);
        MyArrayList<MessageQueue.IdleHandler> myIdleHandlerArrayList = new MyArrayList<>();
        field.set(mainQueue, myIdleHandlerArrayList);
        idleHandlerLagHandlerThread.start();
        idleHandlerLagHandler = new Handler(idleHandlerLagHandlerThread.getLooper());
    } catch (Throwable t) {
        MatrixLog.e(TAG, "reflect idle handler error = " + t.getMessage());
    }
}

这里也非常的细节,是在确定反射成功后才启动在前面创建的 HandlerThread ,其他的逻辑就是使用 MyArrayList 将 MessageQueue 中的 mIdleHandlers 替换一下 ,接下来继续看一下 MyArrayList 的封装逻辑

typescript 复制代码
static class MyArrayList<T> extends ArrayList {
    Map<MessageQueue.IdleHandler, MyIdleHandler> map = new HashMap<>();

    @Override
    public boolean add(Object o) {
        if (o instanceof MessageQueue.IdleHandler) {
            MyIdleHandler myIdleHandler = new MyIdleHandler((MessageQueue.IdleHandler) o);
            map.put((MessageQueue.IdleHandler) o, myIdleHandler);
            return super.add(myIdleHandler);
        }
        return super.add(o);
    }

    @Override
    public boolean remove(@Nullable Object o) {
        if (o instanceof MyIdleHandler) {
            MessageQueue.IdleHandler idleHandler = ((MyIdleHandler) o).idleHandler;
            map.remove(idleHandler);
            return super.remove(o);
        } else {
            MyIdleHandler myIdleHandler = map.remove(o);
            if (myIdleHandler != null) {
                return super.remove(myIdleHandler);
            }
            return super.remove(o);
        }
    }
}

在 MyArrayList 添加数据时,判断如果传入的数据类型是 MessageQueue.IdleHandler ,那么则使用 MyIdleHandler 包装一层,Matrix 玩这一套是真的6,在源码中随处可见的都是这种原理 ,再去看看 MyIdleHandler 都干了啥

java 复制代码
static class MyIdleHandler implements MessageQueue.IdleHandler {
    private final MessageQueue.IdleHandler idleHandler;

    MyIdleHandler(MessageQueue.IdleHandler idleHandler) {
        this.idleHandler = idleHandler;
    }

    @Override
    public boolean queueIdle() {
        idleHandlerLagHandler.postDelayed(idleHandlerLagRunnable, traceConfig.idleHandlerLagThreshold);
        boolean ret = this.idleHandler.queueIdle();
        idleHandlerLagHandler.removeCallbacks(idleHandlerLagRunnable);
        return ret;
    }
}

MyIdleHandler 中就是在开始转发任务前,先发送一个延时消息,在执行结束后移除这个定时消息,如果没有移除则触发这个消息的上报机制

到了这里整个工作就分析完成了

相关推荐
pedestrian_h3 小时前
操作系统-线程
android·java·开发语言
gfdgd xi4 小时前
GXDE 内核管理器 1.0.1——修复bug、支持loong64
android·linux·运维·python·ubuntu·bug
美狐美颜sdk4 小时前
跨平台直播美颜sdk集成攻略:Android、iOS与Web的统一方案
android·前端·ios
❀͜͡傀儡师5 小时前
二维码/条码识别、身份证识别、银行卡识别、车牌识别、图片文字识别、黄图识别、驾驶证(驾照)识别
android·scanner
生莫甲鲁浪戴6 小时前
Android Studio新手开发第三十五天
android·ide·android studio
qq_717410016 小时前
FAQ20472:相机录像镜像功能实现
android
非专业程序员Ping8 小时前
HarfBuzz 实战:五大核心API 实例详解【附iOS/Swift实战示例】
android·ios·swift
流星魂小七8 小时前
颜色选择器
android·着色器·环形颜色选择器·圆形颜色选择器·colorpicker·colorwheelview
cdming9 小时前
LIUNX 与手机安卓的文件互传 的常用方法
android·智能手机
雨白10 小时前
Flow 的异常处理与执行控制
android·kotlin