PHP系统编程--02.PHP守护进程化

什么是守护进程?

一个守护进程一般补认为是一个不对终端进行控制的后台任务。它有三个很显著的特征:在后台运行,与启动他的进程脱离,无须控制终端。经常使用的实现方式是fork() -> setsid() -> fork()php

在glibc里有一个函数daemon。调用此函数,就可以使当前进程脱离终端变成一个守护进程,具体内容参见man daemon。PHP中暂时没有此函数,PHP程序实现守护进程化有2种方法:session

使用系统命令nohup

nohup php myprog.php > log.txt &函数

&,这样执行程序虽然也是转为后台运行,但其实是依赖终端的,当用户退出终端时进程就会被杀掉。须要使用nohup来实现code

PHP脚本函数实现:

<?php
function daemon()
{
    $pid = pcntl_fork();
    if($pid < 0){
        die("fork(1) failed!\n");
    }elseif($pid > 0){
        exit; //让终端启动的进程退出
    }
     chdir("/"); //改变当前目录为根目录
    umask(0); //重设文件权限掩码
    //创建一个有别于终端的新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; //父进程退出, 剩下子进程成为最终的独立进程
    }
    //关闭标准I/O流
    if (defined('STDIN')) {
        fclose(STDIN);
    }
    if (defined('STDOUT')){
        fclose(STDOUT);
    }
    if (defined('STDERR')) {
        fclose(STDERR);
    }
}
daemon();
sleep(1000);

这里较为关键的二个php函数是pcntl_fork()和posix_setsid()继承

  • fork()一个进程,则表示建立了一个运行进程的副本,副本被认为是子进程,而原始进程被认为是父进程。当fork()运行以后,则能够脱离启动他的进程与终端控制等,也意味着父进程能够自由退出。
  • setsid(),它首先使新进程成为一个新会话的“领导者”,最后使该进程再也不控制终端,这也是成为守护进程最关键的一步,这意味着,不会随着终端关闭而强制退出进程。对于一个不会被中断的常驻进程来讲,这是很关键的一步。
  • 进行最后一次fork(),这一步不是必须的,但一般都这么作,它最大的意义是防止得到控制终端。(在直接打开一个终端设备,并且没有使用O_NOCTTY标志的状况下, 会得到控制终端)

其它事项说明:进程

  • chdir() 守护进程默认继承了父进程的当前工做目录,当系统磁盘发生umount时将形成诸多的麻烦,一般将”/” 做为守护进程的当前工做目录,能够避免上述的问题
  • umask() 守护进程默认继承了父进程的文件权限掩码,这就给该子进程使用文件带来了诸多的麻烦。所以,把文件权限掩码设置为0,能够大大加强该守护进程的灵活性
  • fclose(STDIN), fclose(STDOUT), fclose(STDERR) 关闭标准I/O流。用fork函数新建的子进程会从父进程那里继承一些已经打开了的文件。这些被打开的文件可能永远不会被守护进程读写,但它们同样消耗系统资源,并且可能致使所在的文件系统没法卸下。



相关文章
相关标签/搜索