You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

121 lines
3.5 KiB

---
应用进程的启动
---
1. 了解 Linux 下进程启动的方式
2. 熟悉应用进程启动的基本流程
3. 深入理解应用进程启动的原理
#### 进程启动方式
```c
if((pid = fork()) < 0){
// error
}else if(pid == 0){
// child process
// 子进程共享父进程的资源
}else {
// parent process
}
```
```c
if((pid = fork()) < 0){
// error
}else if(pid == 0){
// child process
// 子进程资源完全被 path 替换掉了,这个 path 是个二进程程序的路径
execve(path,argv,env);
}else {
// parent process
}
```
#### 应用进程启动原理
1. 什么时候触发的进程启动?谁发起的?
2. 进程是谁启动的?怎么启动的?
**触发进程启动**
应用进程的启动是被动的,即在启动组件的时候,如果发现该组件所在的进程未被启动,那么这时候就会去启动进程。该操作是 Framework 层去做的:
```java
ProcessRecord app = getProcessRecordLocked(...) {
// app.thread 是 IApplicationThread,是应用层的 binder 句柄
// 为了是 AMS 和应用进行双向调用
if(app != null && app.thread != null) {
// 进程已经启动了,这里就可以启动组件了
return;
}
startProcessLocked(r.processName);
}
```
应用进程启动后的入口函数:
```java
public static void main(String[] args) {
Looper.prepareMainLooper();
ActivityThread thread = new ActivityThread();
thread.attach(false);
Looper.loop();
throw new RuntimeException("...");
}
```
```java
// 获取 AMS 的 binder 对象
final IActivityManager mgr = ActivityManagerNative.getDefault();
mgr.attachApplication(mAppThread);
```
对于 AMS 来说,一个应用进程怎么样才算是启动完成呢?
1. AMS 向 Zygote 发起启动应用进程的请求,然后 Zygote 启动并返回 pid 给 AMS
2. 应用进程启动完成之后,告诉 AMS 注册 ApplicationThread
#### 进程是怎么启动的?
启动进程是在 AMS 里面的 startProcessLocked 函数,这个函数不会直接去启动进程,而是只是发起启动进程:
1. 打开本地 socket,openZygoteSocketIfNeeded 函数
2. 通过 Socket 发送参数列表,参数包括应用进程启动之后要执行的类名,zygoteSendArgsAndGetResult 函数
3. 发送完参数列表之后等待返回创建的进程的 pid 即可
那么 Zygote 里面是如何处理 Socket 请求的?
Zygote 会在一个 for 循环里面不断看有没有消息过来,如果有消息过来就执行 runOnce 函数:
```java
boolean runOnce() {
// 读取 Zygote 发送过来的参数列表
String[] args = readArgumentList();
// 创建应用进程
int pid = Zygote.forkAndSpecialize(...);
if(pid == 0) {
// 子进程,执行 ActivityThread#main 函数
handleChildProc(...);
return true;
} else {
// 父进程把 pid 通过 Socket 在写回去
return handlePArentProc(pid, ...);
}
}
```
#### 课堂作业
1. 为什么是 zygote 来创建进程,而不是 systemserver?
2. AMS 和 zygote 为什么不用 binder 通信?
#### 总结
应用进程是怎么启动的呢?
1. 应用进程是什么时候起的?
2. 由 ASM 向 zygote 发起的,通过 socket 通信
3. zygote fork 出应用进程,执行 ActivityThread#main 入口函数,这个入口函数的类名是 AMS 通过 Socket 发给 zygote 的
4. 进程启动之后要向 AMS 报告,注册 ApplicationThread,这样整个启动才算结束,因为 AMS 有了该报告,应用进程才算是可用,AMS 才能继续启用应用的各个组件