从字面意思这个 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 中就是在开始转发任务前,先发送一个延时消息,在执行结束后移除这个定时消息,如果没有移除则触发这个消息的上报机制
到了这里整个工作就分析完成了