linux php 进程进阶(六)进程组
进程组
进程组,就是一个或是多个进程的集合,每个进程都有一个标识组ID,表示该进程属于哪个进程组。
一般情况下,一个进程组是由一个进程 fork 出来的,之后,它的子进程再去 fork ,最后,得到了一个进程组。当然,单个的进程也是一个进程组。进程组有进程组 id ,它通常是第一个进程的 pid ,也就是组长进程的 pid (以识别进程组)。
创建进程组的目的是用于简化向组内所有进程发送信号的操作,即如果一个信号是发给一个进程组,则这个组内的所有进程都会受到该信号【方便管理】。
进程组的领导进程的PID成为进程组ID。领导进程可以先终结。此时进程组依然存在,并持有相同的PGID,直到进程组中最后一个进程终结。
bash进程启动之后,它会自己调用
setsid
系统函数把自己设置为会话首进程,也会设置自己为组长进程
// 获取当前进程pid
$pid = posix_getpid();
// 打印当前进程pid,父进程pid,进程组pgid,会话sid
fprintf(STDOUT,"pid=%d,ppid=%d,pgid=%d,sid=%d\n",$pid,posix_getppid(),posix_getpgid($pid),posix_getsid($pid));
执行上述代码后可以发现由于 /bin/bash
进程启动时会调用setsid
函数把自己设置为会话首进程,也会设置自己为组长进程,所有看到执行demo1.php 进程的父进程是29228 因为demo1.php在bash进程下执行,会话进程也是29228,使用echo $$
打印当前 bash 进程pid 为 29228.
为什么pgid 不等于29228 呢?
上面我们说过/bin/bash
启动后会调用 setsid
函数设置自己为会话首进程和组长进程,但是自己脚本显示的组长进程却不是/bin/bash
的pid,这是因为脚本执行会调用 setpgid
函数把当前进程设置为组进程,所以显示的pgid不是29228 而是 2893
php 设置进程组
<?php
function show(){
$pid = posix_getpid();
fprintf(STDOUT,"pid=%d,ppid=%d,pgid=%d,sid=%d\n",$pid,posix_getppid(),posix_getpgid($pid),posix_getsid($pid));
}
show();
$map_pid = [];
$pid = pcntl_fork();
if($pid > 0){
$map_pid[$pid] = $pid;
$pid = pcntl_fork();
if($pid >0){
$map_pid[$pid] = $pid;
}else{
$ppid = posix_getpid();
posix_setpgid($pid,$pid);
$ppid = pcntl_fork();
if($ppid > 0){
// 子进程退出回收资源,防止孤儿进程
pcntl_waitpid(-1, $status);
}
}
}
show();
if($pid > 0){
while(1){
// 子进程退出回收资源,防止孤儿进程
$pid = pcntl_waitpid(-1, $status);
if($pid > 0){
print("pid= $pid 已经结束\n");
unset($map_pid[$pid]);
}
// 子进程退出完后,父进程退出
if(empty($map_pid)){
break;
}
}
}
posix_setpgid 函数,设置进程为组长进程,需要传入两个参数,第一个是进程pid,第二个是要设置为组长进程的pid
执行demo2.php 后我们可以看到生成了4个进程,19502进程设置自己为组长进程后fork子进程,子进程pid为19503 ,组id为19502 也就是父进程id