BlockCanary分析实现原理

2019-12-30  本文已影响0人  only_run

介绍

BlockCanary 一个android UI检测工具,追踪view耗时操作;
附上地址:
https://github.com/markzhai/AndroidPerformanceMonitor
http://blog.zhaiyifan.cn/2016/01/16/BlockCanaryTransparentPerformanceMonitor/

核心代码

//BlockCanary
    public void start() {
        if (!mMonitorStarted) {
            mMonitorStarted = true;
            Looper.getMainLooper().setMessageLogging(mBlockCanaryCore.monitor);
        }
    }

监听loop处理事件

原来是监听主线程的 loop事件;看下怎样监听的

 public static void loop() {
      //...
        for (;;) {
            Message msg = queue.next(); // might block
            if (msg == null) {
                // No message indicates that the message queue is quitting.
                return;
            }

            // This must be in a local variable, in case a UI event sets the logger
            final Printer logging = me.mLogging;
            if (logging != null) {//1
                logging.println(">>>>> Dispatching to " + msg.target + " " +
                        msg.callback + ": " + msg.what);
            }
        //...
            try {
                msg.target.dispatchMessage(msg);//2
                dispatchEnd = needEndTime ? SystemClock.uptimeMillis() : 0;
            } finally {
                if (traceTag != 0) {
                    Trace.traceEnd(traceTag);
                }
            }
         //..

            if (logging != null) {//3
                logging.println("<<<<< Finished to " + msg.target + " " + msg.callback);
            }
  //...
        }
    }

原来主线程的消息队列抽取消息 去处理的前后,注释1,注释2 都会执行 logging的日志打印;而logging正是 BlockCanary传入的,实现类是
LooperMonitor;

class LooperMonitor implements Printer {
 @Override
    public void println(String x) {
        if (mStopWhenDebugging && Debug.isDebuggerConnected()) {
            return;
        }
        if (!mPrintingStarted) {//1
            mStartTimestamp = System.currentTimeMillis();
            mStartThreadTimestamp = SystemClock.currentThreadTimeMillis();
            mPrintingStarted = true;
            startDump();
        } else {//2
            final long endTime = System.currentTimeMillis();
            mPrintingStarted = false;
            if (isBlock(endTime)) {//3
                notifyBlockEvent(endTime);
            }
            stopDump();
        }
    }
  
    private boolean isBlock(long endTime) {
        return endTime - mStartTimestamp > mBlockThresholdMillis;
    }
}

注释1 记录当前时间和开始标记,代码执行到注释2的时候,自然是要记录结束时间;
执行注释3 判断handler的消息处理是否 超过阀值,如果超出 就给出ui卡顿提示 以及方法调用(stacktrace)和cpu使用信息,来帮助开发者 即使发现ui卡顿问题;

BlockCanary如何读取stacktrace和cpu使用信息

BlockCanary 有两个采样器,StackSampler和CpuSampler;
handler开始处理消息时,执行startDump;采样器开始工作
handler开始处理结束时,执行stopDumo;采样器结束工作

在handle开始处理信息时,使用Stack采样器获取方法调用信息 使用Cpu采样器获取Cpu使用信息;
在handle处理信息完毕时,采样器结束读取。
这样就获取到了handler处理消息过程中的 方法调用信息和cpu使用信息;

附StackSampler,CpuSampler 部分代码

class StackSampler extends AbstractSampler {
   @Override
    protected void doSample() {
        StringBuilder stringBuilder = new StringBuilder();

        for (StackTraceElement stackTraceElement : mCurrentThread.getStackTrace()) {
            stringBuilder
                    .append(stackTraceElement.toString())
                    .append(BlockInfo.SEPARATOR);
        }

        synchronized (sStackMap) {
            if (sStackMap.size() == mMaxEntryCount && mMaxEntryCount > 0) {
                sStackMap.remove(sStackMap.keySet().iterator().next());
            }
            sStackMap.put(System.currentTimeMillis(), stringBuilder.toString());
        }
    }
}

class CpuSampler extends AbstractSampler {
    @Override
    protected void doSample() {
        BufferedReader cpuReader = null;
        BufferedReader pidReader = null;

        try {
            cpuReader = new BufferedReader(new InputStreamReader(
                    new FileInputStream("/proc/stat")), BUFFER_SIZE);
            String cpuRate = cpuReader.readLine();
            if (cpuRate == null) {
                cpuRate = "";
            }

            if (mPid == 0) {
                mPid = android.os.Process.myPid();
            }
            pidReader = new BufferedReader(new InputStreamReader(
                    new FileInputStream("/proc/" + mPid + "/stat")), BUFFER_SIZE);
            String pidCpuRate = pidReader.readLine();
            if (pidCpuRate == null) {
                pidCpuRate = "";
            }

            parse(cpuRate, pidCpuRate);
        } catch (Throwable throwable) {
            Log.e(TAG, "doSample: ", throwable);
        } finally {
            try {
                if (cpuReader != null) {
                    cpuReader.close();
                }
                if (pidReader != null) {
                    pidReader.close();
                }
            } catch (IOException exception) {
                Log.e(TAG, "doSample: ", exception);
            }
        }
    }
}

StackSampler和CpuSampler都有一个HandlerThread对象,用来在各自的子线程中 间隔性的启动任务;来读取 主线程方法调用和cpu使用信息。

上一篇 下一篇

猜你喜欢

热点阅读