美文网首页
php实现守护进程

php实现守护进程

作者: weylau | 来源:发表于2019-08-29 11:37 被阅读0次

    何为守护进程

    • 守护进程是在后台运行不受终端控制的进程(如输入、输出等)
      守护进程脱离终端的主要原因有两点:
    • 用来启动守护进程的终端在启动守护进程之后,需要执行其他任务。
    • (如其他用户登录该终端后,以前的守护进程的错误信息不应出现)由终端上的一些键所产生的信号(如中断信号),不应对以前从该终端上启动的任何守护进程造成影响

    创建守护进程的过程

      1. 调用fork创建子进程。父进程终止,让子进程在后台继续执行。
      1. 子进程调用setsid产生新会话期并失去控制终端调用setsid()使子进程进程成为新会话组长和新的进程组长,同时失去控制终端。
      1. 忽略SIGHUP信号。会话组长进程终止会向其他进程发该信号,造成其他进程终止。
      1. 调用fork再创建子进程。子进程终止,子子进程继续执行,由于子子进程不再是会话组长,从而禁止进程重新打开控制终端。
      1. 改变当前工作目录为根目录。一般将工作目录改变到根目录,这样进程的启动目录也可以被卸掉。
      1. 关闭打开的文件描述符,打开一个空设备,并复制到标准输出和标准错误上。 避免调用的一些库函数依然向屏幕输出信息。
      1. 重设文件创建掩码清除从父进程那里继承来的文件创建掩码,设为0。
      1. 用openlog函数建立与syslogd的连接。

    php实现守护进程实例

    <?php
    //php代码实现守护进程
    function daemon()
    {
        $pid = pcntl_fork();
        if ($pid < 0) {
            die("fork(1) failed!\n");
        } elseif ($pid > 0) {
            //1.父进程直接退出
            exit;
        }
        //执行到这里就是子进程
        //2.建立一个有别于终端的新session以脱离终端
        $sid = posix_setsid();
        if (!$sid) {
            die("setsid failed!\n");
        }
        //这一部不是必须的
        $pid = pcntl_fork();
        if ($pid < 0) {
            die("fork(1) failed!\n");
        } elseif ($pid > 0) {
            exit; //父进程退出, 剩下子进程成为最终的独立进程
        }
        //3.设置当前进程的工作目录为根目录,不依赖于其他
        chdir("/");
        //4.umask设置为0确保将来进程有最大的文件操作权限
        umask(0);
        //5.关闭标准I/O流
        if (defined('STDIN')) {
            fclose(STDIN);
        }
        if (defined('STDOUT')) {
            fclose(STDOUT);
        }
        if (defined('STDERR')) {
            fclose(STDERR);
        }
    }
    
    daemon();
    //设置进程名
    cli_set_process_title("DaemonTest");
    $redis = new Redis();
    $redis->connect('127.0.0.1', 6381);
    while (true) {
        //echo 1; 不要任何输出echo 因为标准输入流关闭了,会异常导致进程终止
        $redis->set("name", "lemon" . mt_rand());
        sleep(3);
    }
    
    /**
     * 注意:
     * 这里较为关键的二个php函数是pcntl_fork()和posix_setsid()
     *
     * fork()一个进程,则表示创建了一个运行进程的副本,副本被认为是子进程,而原始进程被认为是父进程。当fork()运行之后,则可以脱离启动他的进程与终端控制等,也意味着父进程可以自由退出。
     * setsid(),它首先使新进程成为一个新会话的“领导者”,最后使该进程不再控制终端,这也是成为守护进程最关键的一步,这意味着,不会随着终端关闭而强制退出进程。对于一个不会被中断的常驻进程来说,这是很关键的一步。
     * 进行最后一次fork(),这一步不是必须的,但通常都这么做,它最大的意义是防止获得控制终端。(在直接打开一个终端设备,而且没有使用O_NOCTTY标志的情况下, 会获得控制终端)
     * 其它事项说明:
     *
     * chdir() 守护进程默认继承了父进程的当前工作目录,当系统磁盘发生umount时将造成诸多的麻烦,通常将”/” 作为守护进程的当前工作目录,可以避免上述的问题
     * umask() 守护进程默认继承了父进程的文件权限掩码,这就给该子进程使用文件带来了诸多的麻烦。因此,把文件权限掩码设置为0,可以大大增强该守护进程的灵活性
     * fclose(STDIN), fclose(STDOUT), fclose(STDERR)
     * 关闭标准I/O流。用fork函数新建的子进程会从父进程那里继承一些已经打开了的文件。这些被打开的文件可能永远不会被守护进程读写,但它们一样消耗系统资源,而且可能导致所在的文件系统无法卸下。
     */
    

    启动守护进程

    $ php daemon.php
    

    查看进程

    # ps -ef | grep Test
    root      18838      1  0 11:20 ?        00:00:00 DaemonTest
    

    相关文章

      网友评论

          本文标题:php实现守护进程

          本文链接:https://www.haomeiwen.com/subject/bmzbectx.html