Activity Task实现对Activity的性能监控。我们在“开始”一篇里提到过,Activity的性能采集实现方式有两种,Instrumentation和AOP,而ActivitTask的start方法主要就做了一个判断:如果使能了ApmTask.FLAG_COLLECT_ACTIVITY_INSTRUMENTATION,也就是开启了Instrumentation方式,但是InstrumentationHooker失败,mIsCanWork要设置为false,也就是mIsCanWork不能work。
我们接下来对两种采集方式分别进行分析。
我们先来看看Instrumentation方式。这是一种在运行期进行hook的方式。从ActivityTask的start方法中,我们看到Activity 的监控,并不是在ActivityTask的start方法中开启的。那么,对Activity的监控,是在哪里开启的呢?其实我们在“开始”一篇中,已经分析过了,实际上,这是在TaskManager类的startWorkTasks方法中。这个方法之前分析过了,这里不再详述。这个方法内会决定用Instrumentation方式还是用AOP方式进行Activity信息的收集,并且会启动相应的任务。对于Instrumentation方式,会通过InstrumentationHooker.doHook()方法来启动hook。这个方法又调用本类的hookInstrumentation,如果调用成功,会把isHookSucceed设置为true。hookInstrumentation的代码是这样的:
private static void hookInstrumentation() throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, IllegalAccessException, NoSuchFieldException {
Class> c = Class.forName("android.app.ActivityThread");
Method currentActivityThread = c.getDeclaredMethod("currentActivityThread");
boolean acc = currentActivityThread.isAccessible();
if (!acc) {
currentActivityThread.setAccessible(true);
}
Object o = currentActivityThread.invoke(null);
if (!acc) {
currentActivityThread.setAccessible(acc);
}
Field f = c.getDeclaredField("mInstrumentation");
acc = f.isAccessible();
if (!acc) {
f.setAccessible(true);
}
Instrumentation currentInstrumentation = (Instrumentation) f.get(o);
Instrumentation ins = new ApmInstrumentation(currentInstrumentation);
f.set(o, ins);
if (!acc) {
f.setAccessible(acc);
}
}
通过反射找到android.app.ActivityThread类,再找到类中的currentActivityThread的方法。判断方法是否能访问,如果不能,首先要激活访问。通过invoke调用这个方法之后,如果初始方法是不可访问的,需要恢复为不可访问。找到ActivityThread类的mInstrumentation属性,同样,如果不可访问,需要打开访问权限,不详述。通过Field的get方法,得到currentActivityThread的mInstrumentation,currentInstrumentation。通过currentInstrumentation构建一个ApmInstrumentation的对象ins,通过Field的set方法,设置currentActivityThread的mInstrumentation为新的ApmInstrumentation类型的ins。最后,不要忘了恢复field的可访问状态。
private static class FirstFrameRunnable implements Runnable {
private Activity activity;//Activity
private int startType;//启动类型
private long startTime;//开始时间
public FirstFrameRunnable(Activity activity, int startType, long startTime) {
this.startTime = startTime;
this.activity = activity;
this.startType = startType;
}
@Override
public void run() {
if (DEBUG) {
LogX.d(TAG, SUB_TAG, "FirstFrameRunnable time:" + (System.currentTimeMillis() - startTime));
}
if ((System.currentTimeMillis() - startTime) >= ArgusApmConfigManager.getInstance().getArgusApmConfigData().funcControl.activityFirstMinTime) {
saveActivityInfo(activity, startType, System.currentTimeMillis() - startTime, ActivityInfo.TYPE_FIRST_FRAME);
}
//保存应用冷启动时间
if (DEBUG) {
LogX.d(TAG, SUB_TAG, "FirstFrameRunnable time:" + String.format("[%s, %s]", ActivityCore.isFirst, ActivityCore.appAttachTime));
}
if (ActivityCore.isFirst) {
ActivityCore.isFirst = false;
if (ActivityCore.appAttachTime <= 0) {
return;
}
int t = (int) (System.currentTimeMillis() - ActivityCore.appAttachTime);
AppStartInfo info = new AppStartInfo(t);
ITask task = Manager.getInstance().getTaskManager().getTask(ApmTask.TASK_APP_START);
if (task != null) {
task.save(info);
if (AnalyzeManager.getInstance().isDebugMode()) {
AnalyzeManager.getInstance().getParseTask(ApmTask.TASK_APP_START).parse(info);
}
} else {
if (DEBUG) {
LogX.d(TAG, SUB_TAG, "AppStartInfo task == null");
}
}
}
}
}
它有三个属性,分别是当前Activity activity, 启动类型startType,和启动时间startTime。重点看一下它的run方法。如果当前系统时间减去startTime不小于activityFirstMinTime,就保存Activity的信息。activityFirstMinTime是设定的Activity第一帧最小时间。如果Activity是第一次启动,还要记录冷启动时间。如果ActivityCore.appAttachTime小于或者等于0(应用没有启动),返回不处理。用当前系统时间减去ActivityCore.appAttachTime,这个就是冷启动时间。获取正在运行的ApmTask.TASK_APP_START,保存task的信息。
其它的几个Activity生命周期的回调方法都是类似的。
如果我们没有配置ApmTask.FLAG_COLLECT_ACTIVITY_INSTRUMENTATION,那么默认就会以aop的方式对Activity进行数据采集。具体的采集方式,我们在分析argus-apm-aop方式时,已经讨论过了。