Binder是支持并发请求的,也就是多个客户进程可以同时向同一个服务进程发送请求,服务进程需要同时处理这些请求。
服务进程使用线程池接收和处理请求,线程池是管理多线程的一种方式,使用线程池有以下3点好处。
- 提升系统性能:线程池可重复使用已经创建好的线程,降低了创建和销毁线程带来的系统开销。
- 提高响应速度:线程池会提前准备好线程,任务到达时可立刻处理。
- 合理使用资源:线程不能太多,太多的空闲线程会浪费系统资源。也不能太少,当业务繁忙时线程太少会影响速度。 而线程池可根据业务的繁忙程度动态调整线程的数量,合理利用系统资源满足业务需求。
服务进程开启线程池的方法比较简单,方法如下
int main() { .... ProcessState::self()->startThreadPool(); IPCThreadState::self()->joinThreadPool(); } |
startThreadPool开启一个新线程,并加入到线程池中接收和处理请求。joinThreadPool将主线程加入到线程池。经过以上2步,线程池就有2个空闲线程等待处理请求。
如果某个时间点服务进程同时收到多个请求,当前的线程已经处理不过来,会动态地增加线程,增加新线程的条件如下。
static int binder_thread_read(...) { if(proc->requested_threads + proc->ready_threads == 0 && proc->requested_threads_started < proc->max_threads && (thread->looper & (BINDER_LOOPER_STATE_REGISTERED | BINDER_LOOPER_STATE_ENTERED)) ) { proc->requested_threads++; if (put_user(BR_SPAWN_LOOPER, (uint32_t __user *)buffer)) return -EFAULT; } } |
创建新线程需要满足以下4个条件。
- 当前没有申请新线程:request_threads表示当前正在请求创建线程的数量,request_threads不为0说明当前有正在创建的线程,不满足创建新线程的条件。
- 当前没有空闲线程:ready_threads表示当前空闲线程的数量,空闲线程指处于阻塞等待状态的线程。ready_threads不为0说明当前的业务不繁忙,没必再创建新的线程。
- 当前线程数量没有达到最大值:每个线程都会占用系统资源,不能无限制地创建线程,线程数量达到最大值后不再增加。
- 当前线程处于循环:只有处于循环的线程才能创建新线程,只是发请求的线程是不能创建新线程的。
满足后返回命令BR_SPAWN_LOOPER,用户进程收到该命令后开始创建新线程。
以上内容参考 《Android图形显示系统》作者 李先儒,,清华大学出版社