这里介绍一些Android的一些冷知识。
App进程是从哪里开始启动的
ActivityManagerService类的源码在Sdk里有,是Android中最核心的服务之一,负责系统中四大组件的启动、切换、调度及应用进程的管理和调度等等工作。这个类是运行在系统进程里,由系统管理,通过跨进程通信的方式与我们的App进行通信。这个类有个方法
private final void startProcessLocked(ProcessRecord app, String hostingType, String hostingNameStr, String abiOverride, String entryPoint, String[] entryPointArgs) {
......
Process.ProcessStartResult startResult = Process.start(entryPoint,
app.processName, uid, uid, gids, debugFlags, mountExternal,
app.info.targetSdkVersion, app.info.seinfo, requiredAbi, instructionSet,
app.info.dataDir, entryPointArgs);
......
}
我们的App进程最初就是由这个方法创建的。
App的入口方法在哪里?
任何语言写的任何程序都会有执行入口。Android App的入口是ActivityThread的main方法。具体定义是在ActivityManagerService的startProcessLocked方法里。
private final void startProcessLocked(ProcessRecord app, String hostingType, String hostingNameStr, String abiOverride, String entryPoint, String[] entryPointArgs) {
......
// Start the process. It will either succeed and return a result containing
// the PID of the new process, or else throw a RuntimeException.
boolean isActivityProcess = (entryPoint == null);
if (entryPoint == null) entryPoint = "android.app.ActivityThread";
Trace.traceBegin(Trace.TRACE_TAG_ACTIVITY_MANAGER, "Start proc: " +
app.processName);
checkTime(startTime, "startProcess: asking zygote to start proc");
Process.ProcessStartResult startResult = Process.start(entryPoint,
app.processName, uid, uid, gids, debugFlags, mountExternal,
app.info.targetSdkVersion, app.info.seinfo, requiredAbi, instructionSet,
app.info.dataDir, entryPointArgs);
checkTime(startTime, "startProcess: returned from zygote!");
Trace.traceEnd(Trace.TRACE_TAG_ACTIVITY_MANAGER);
......
}
if (entryPoint == null) entryPoint = "android.app.ActivityThread";这一行定义了App的入口类。
App创建Activity的地方在哪里
创建Activity的消息是从ActivityManagerService通过跨进程通信的方式发送到
-> ApplicationThread
-> scheduleLaunchActivity
mH->handleMessage -> LAUNCH_ACTIVITY
-> handleLaunchActivity
-> performLaunchActivity
-> mInstrumentation.newActivity
-> (Activity)cl.loadClass(className).newInstance();
最终通过反射的方式创建出了Activity实例。
Activity实例引用被谁持有?
Activity创建了实例对象后,肯定会被某个对象持有,否则引用计数为0后,Activity实例就会被GC回收掉。
private Activity performLaunchActivity(ActivityClientRecord r, Intent customIntent) {
......
Activity activity = null;
......
activity = mInstrumentation.newActivity(cl,component.getClassName(), r.intent);
......
r.activity = activity;
mActivities.put(r.token, r);
......
}
Activity在创建之后就被mActivities持有。mActivities被ActivityThread持有。ActivityThread对象是在App的入口main方法实例化的。main方法除非进程被杀死否则main是不会结束的。
public static void main(String[] args) {
......
ActivityThread thread = new ActivityThread();
......
}
Activity实例什么时候被销毁?
前面说到Activity被ActivityThread的mActivities持有。那什么时候Activity实例被销毁呢?说到销毁大家可能会想到Activity生命周期的onDestory方法。
-> ApplicationThread
-> scheduleDestroyActivity
mH->handleMessage -> DESTROY_ACTIVITY
-> handleDestroyActivity
-> performDestroyActivity
// 从这里继续会执行到Activity的onDestory方法
-> mInstrumentation.callActivityOnDestroy(r.activity);
// Activity实例从mActivities被移除
-> mActivities.remove(token);
当Activity实例从mActivities里移除后,如果当前Activity实例没有被其他对象强引用或者说Activity没有内存泄漏,那么下次GC的时候Activity实例就会被销毁掉。