// 设置umask为0,这样,当前进程创建的文件权限则为777
umask( 0 );
$pid = pcntl_fork();
if( $pid < 0 ){
exit('fork error.');
} else if( $pid > 0 ) {
// 主进程退出
exit();
}
// 子进程继续执行
// 最关键的一步来了,执行setsid函数!
if( !posix_setsid() ){
exit('setsid error.');
}
// 理论上一次fork就可以了
// 但是,二次fork,这里的历史渊源是这样的:在基于system V的系统中,通过再次fork,父进程退出,子进程继续
// 保证形成的daemon进程绝对不会成为会话首进程,不会拥有控制终端。
$pid = pcntl_fork();
if( $pid < 0 ){
exit('fork error');
} else if( $pid > 0 ) {
// 主进程退出
exit;
}
// 子进程继续执行
// 给进程重新起个名字
cli_set_process_title('php master process');
// 由于*NIX好像并没有(如果有,请告知)可以获取父进程fork出所有的子进程的ID们的功能,所以这个需要我们自己来保存
$child_pid = [];
// 父进程安装SIGCHLD信号处理器并分发
pcntl_signal( SIGCHLD, function(){
// 这里注意要使用global将child_pid全局化,不然读到去数组将为空,具体原因可以自己思考下
global $child_pid;
// 如果子进程的数量大于0,也就说如果还有子进程存活未 退出,那么执行回收
$child_pid_num = count( $child_pid );
if( $child_pid_num > 0 ){
// 循环子进程数组
foreach( $child_pid as $pid_key => $pid_item ){
$wait_result = pcntl_waitpid( $pid_item, $status, WNOHANG );
// 如果子进程被成功回收了,那么一定要将其进程ID从child_pid中移除掉
if( $wait_result == $pid_item || -1 == $wait_result ){
unset( $child_pid[ $pid_key ] );
}
}
}
} );
// fork出5个子进程出来,并给每个子进程重命名
for( $i = 1; $i <= 5; $i++ ){
$_pid = pcntl_fork();
if( $_pid < 0 ){
exit();
} else if( 0 == $_pid ) {
// 重命名子进程
cli_set_process_title('php worker process');
// 啦啦啦啦啦啦啦啦啦啦,请在此处编写你的业务代码
// do something ...
// 啦啦啦啦啦啦啦啦啦啦,请在此处编写你的业务代码
// 子进程退出执行,一定要exit,不然就不会fork出5个而是多于5个任务进程了
exit();
} else if( $_pid > 0 ) {
// 将fork出的任务进程的进程ID保存到数组中
$child_pid[] = $_pid;
}
}
// 主进程继续循环不断派遣信号
while( true ){
pcntl_signal_dispatch();
// 每派遣一次休眠一秒钟
sleep( 1 );
}