php进程之daemon化正确操作

2018-05-06 22:18:51

daemon 音标 : ['di:mən] , 中文含义为守护神或精灵的意思 . 其实它还有个意思 : 守护进程

守护进程简单地说就是可以脱离终端而在后台运行的进程 . 这在Linux中是非常常见的一种进程 , 比如apache或者mysql等服务启动后 , 就会以守护进程的方式进驻在内存中

以PHP为例 , 假如我有个耗时间的任务需要跑在后台 : 将所有mysql中user表中的2000万用户全部导入到redis中做预热缓存 , 那么这个任务估计一时半会是不会结束的 , 这个时候就需要编写一个php脚本以daemon形式运行在系统中 , 结束后自动退出

在Linux中 , 大概有三种方式实现脚本后台化 :

  1. 在命令后添加一个&符号 , 比如 php task.php & . 这个方法的缺点在于 如果terminal终端关闭 , 无论是正常关闭还是非正常关闭 , 这个php进程都会随着终端关闭而关闭 , 其次是代码中如果有echo或者print_r之类的输出文本 , 会被输出到当前的终端窗口中 .
  2. 使用nohup命令 , 比如 nohup php task.php & . 默认情况下 , 代码中echo或者print_r之类输出的文本会被输出到php代码同级目录的nohup.out文件中 . 如果你用exit命令或者关闭按钮等正常手段关闭终端 , 该进程不会被关闭 , 依然会在后台持续运行 . 但是如果终端遇到异常退出或者终止 , 该php进程也会随即退出 . 本质上 , 也并非稳定可靠的daemon方案 .
  3. 使用fork和setsid , 我暂且称之为 : *nix解决方案 . 具体看下代码 :
<?php
    // 一次fork  
    $pid = pcntl_fork();
    if ( $pid < 0 ) {
      exit( ' fork error. ' );
    } else if( $pid > 0 ) {
      exit( ' parent process. ' );
    }
    // 将当前子进程提升会会话组组长 这是至关重要的一步 
    if ( ! posix_setsid() ) {
      exit( ' setsid error. ' );
    }
    // 二次fork
    $pid = pcntl_fork();
    if( $pid < 0 ){
      exit( ' fork error. ' );
    } else if( $pid > 0 ) {
      exit( ' parent process. ' );
    }
    
    // 真正的逻辑代码们 下面仅仅写个循环以示例
    for( $i = 1 ; $i <= 100 ; $i++ ){
      sleep( 1 );
      file_put_contents( 'daemon.log', $i, FILE_APPEND );
    }
?>

具体步骤

创建子进程,父进程退出
  • 父进程先与子进程退出,子进程则会被1号进程收养,这个子进程就会成为init的子进程
  • php使用pcntl_fork()来创建子进程。
  • pcntl_fork()返回一个整型值,在父进程里面返回的是子进程的id,子进程返回的是0,失败则返回-1。这样我们就可以根据这个来分别控制父进程和子进程执行任务。
子进程创建会话
  • 这个是重要的一步,在这一步中该子进程会做这些事情:
  1. 让进程摆脱原会话的控制;
  2. 让进程摆脱员进程组的控制;
  3. 让进程摆脱终端的控制。

php这里使用posix_setsid()来在这个子进程中创建会话,使得这个进程成为会话组组长

通过这两步骤就可以创建一个守护进程了。如果多需要多进程,那么只需要在这个守护进程fork子进程就可以。