/首页
/开源
/关于
PHP多进程初探 --- 利用多进程开发点儿东西吧
发表@2018-09-13 09:15:21
更新@2023-01-21 22:47:40
##### 干巴巴地叨逼叨了这么久,时候表演真正的技术了! ##### 做个高端点儿的玩意吧,加入我们要做一个任务系统,这个系统可以在后台帮我们完成一大波(注意是一大波)数据的处理,那么我们自然想到,多开几个进程分开处理这些数据,同时我们不能执行了php task.php后终端挂起,万一一不小心关闭了终端都会导致任务失败,所以我们还要实现程序的daemon化。好啦,开始了! ##### 首先,我们第一步就得将程序daemon化了! ```php // 设置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'); ``` ##### 加入我们fork出5个子进程就可以搞定这些任务,那么fork出5个子进程,同时父进程要负责这5个子进程的状态等。 ```php // 由于*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 ); } ```