多进程demo里介绍过,如何fork子进程。 但是,谁家的孩子谁管,你fork了得记得处理啊,不然哪天捣点乱啥的,是吧。 进程与进程之间通信的方式有很多种,这里用最简单的信号通信。当我们kill -9 一个进程的时候,实际上就是向该进程发送一个终止信号,之前已
多进程demo里介绍过,如何fork子进程。
但是,谁家的孩子谁管,你fork了得记得处理啊,不然哪天捣点乱啥的,是吧。
进程与进程之间通信的方式有很多种,这里用最简单的信号通信。当我们kill -9 一个进程的时候,实际上就是向该进程发送一个终止信号,之前已经绑定了信号处理器,所以,我们可以用其他进程通知另一个进程做出相应的反应(kill -9 的时候是退出)。父子进程通信也不例外,父进程绑定信号处理器,用来协调子进程,子进程向父进程发送信号,让父进程有相应的反应。
绑定信号处理器:pcntl_signal http://cn2.php.net/manual/en/function.pcntl-signal.php#70161
发送信号:posix_kill http://cn2.php.net/manual/en/function.posix-kill.php
需要注意一点,信号的不可重入性,当一个子进程的信号处理器正处理时,将忽略其他的信号处理。所以,需要pcntl_watipid
http://cn2.php.net/manual/en/function.pcntl-waitpid.php 来等待。
运用上边的理论,实现了一个固定子进程处理任务的程序:
父进程fork子进程,然后绑定SIGCHLD信号,子进程结束触发父进程的SIGCHLD信号处理器,末尾主循环,统计子进程的个数,当低于指定值时,fork新的子进程:代码如下:
0 ) { if($signal == SIGCHLD){ $child_nums--; } } } function fork_do($c){ global $child_nums; while($c--) { if(pcntl_fork() == 0 ) { real_do(); } else { $child_nums++; } } } function real_do(){ sleep(rand(1,3)); exit(); } $manage_pid = getmypid(); $fork_nums = 4; pcntl_signal(SIGCHLD,'catch_signal'); fork_do($fork_nums); while($child_nums != 0){ echo "Childs " . $child_nums . "\n"; if($child_nums < $fork_nums) { fork_do($fork_nums - $child_nums); } }
注:要看动态效果显得很酷的话,把sleep改成usleep或者干脆去掉吧。
原文地址:php父子进程通信, 感谢原作者分享。