data:image/s3,"s3://crabby-images/74513/74513aa1da402cc390f6054c44bd6d443846ae27" alt="Android进阶解密"
3.3 Binder 线程池启动过程
在3.2.2节中学习了Zygote接收请求并创建应用程序进程,其中有一个遗留的知识点就是,在应用程序进程创建过程中会启动Binder线程池。我们查看ZygoteInit类的zygoteInit方法,如下所示:
data:image/s3,"s3://crabby-images/e28df/e28df7da9f71680ba8d827fd9ab46eb8012205c3" alt=""
在注释1处会在新创建的应用程序进程中创建Binder线程池,下面来查看nativeZygoteInit方法:
data:image/s3,"s3://crabby-images/02e8c/02e8cb3b7f39996faa05aa85bd1706b2ed9b65f5" alt=""
很明显nativeZygoteInit是一个JNI方法,它对应的函数是什么呢?在AndroidRuntime.cpp的JNINativeMethod数组中我们得知它对应的函数是com_android_internal_os_ZygoteInit_nativeZygoteInit,如下所示:
data:image/s3,"s3://crabby-images/52432/5243299d846284922e085796b809e79e59e4d1ee" alt=""
接着来查看com_android_internal_os_ZygoteInit_nativeZygoteInit函数:
data:image/s3,"s3://crabby-images/1bca3/1bca3e1ebfb547b6b71184d7e3dd4505351f3225" alt=""
gCurRuntime是AndroidRuntime类型的指针,它是在AndroidRuntime初始化时就创建的,如下所示:
data:image/s3,"s3://crabby-images/fce12/fce12b96e28c65bc369f0b748ea33b41c3f60382" alt=""
AppRuntime继承自AndroidRuntime,AppRuntime创建时就会调用AndroidRuntime的构造函数,gCurRuntime就会被初始化,它指向的是AppRuntime,我们来查看AppRuntime的onZygoteInit函数,AppRuntime在app_main.cpp中实现,如下所示:
data:image/s3,"s3://crabby-images/8d95b/8d95bc9087f6f5613c24499d8afb88409a69ea65" alt=""
最后一行会调用ProcessState的startThreadPool函数来启动Binder线程池:
data:image/s3,"s3://crabby-images/bb3b7/bb3b7f16ad305a305fbe424cdb8b3c9afbdba6e7" alt=""
支持Binder通信的进程中都有一个ProcessState类,它里面有一个mThreadPoolStarted变量,用来表示Binder线程池是否已经被启动过,默认值为false。在每次调用startThreadPool函数时都会在注释1处先检查这个标记,从而确保Binder线程池只会被启动一次。如果Binder 线程池未被启动,则在注释2处设置mThreadPoolStarted为true,并调用spawnPooledThread函数来创建线程池中的第一个线程,也就是线程池的主线程,如下所示:
data:image/s3,"s3://crabby-images/863eb/863eb486591a94e7a1810943348d623b304ead47" alt=""
可以看到Binder线程为一个PoolThread。在注释1处调用PoolThread的run函数来启动一个新的线程。下面来查看PoolThread类做了什么:
data:image/s3,"s3://crabby-images/a30ff/a30ffc178a3d79bf8f61e20471aa9d75a7b7781d" alt=""
PoolThread类继承了Thread类。在注释1处调用IPCThreadState的joinThreadPool函数,将当前线程注册到Binder驱动程序中,这样我们创建的线程就加入了Binder线程池中,新创建的应用程序进程就支持Binder进程间通信了,我们只需要创建当前进程的Binder对象,并将它注册到ServiceManager中就可以实现Binder进程间通信,而不必关心进程间是如何通过Binder进行通信的。