记录php daemon 进程 遇到的问题--posix_setsid函数

总结一下先~

  • 一个守护进程一般需要root权限,因为可能要使用特殊端口1-1024及其他权限
  • 一个守护进程的父进程会被fork之后被杀掉,所以可以说他的父进程是init进程。
  • 一个守护进程无需交互,也和终端(teriminalsession)无关,所以任何输出,无论是向标准输出还是错误输出,都需要特殊处理,涉及到的就是stdout和stderr

上代码

下面是我用php 编写一个守护进程 demo

好下面我们执行操作,输出如下

root@tb:/home/tb/linuxing# php php_daemon.php 
fork succ
loop0
root@tb:/home/tb/linuxing# loop1
loop2
loop3
loop4
...
loop20

查看demo.txt

cat demo.txt 
0--2016-07-15 17:49:47
1--2016-07-15 17:49:48
2--2016-07-15 17:49:49
...

问题来了

貌似没有问题,但是用以上php代码,执行后,马上关闭当前终端。则发现程序并不会完整输出20行数据,只是部分数据。
问题复现步骤:
1.ubuntu终端A中 执行 php php_daemon.php
2.关闭终端A
3.打开新终端B,ps -aux |grep php 发现无此进程

如果手慢,自己把握时间或者调整for 次数。。

为什么呢

进程从创建它的父进程那里继承了打开的文件描述符。如不关闭,将会浪费系统资源,(这倒是小事),造成进程所在的文件系统无法卸下以及引起无法预料的错误。
所以需要关闭这些
fclose(STDIN),fclose(STDOUT),fclose(STDERR)
关闭标准输入输出与错误显示。

正确代码之一

再解释一下

如果想在关闭当前终端后继续执行
需要关闭echo 那一行,因为当然echo 和当然session关联,sesssion关闭后,echo就会导致php致命错误,所以下面的file_put_contents不会执行。

所以为了避免除显示输出的echo导致php错误的问题,我们一般建议这样

      global $STDOUT, $STDERR;
      fclose(STDOUT);
      fclose(STDERR);
      $STDOUT = fopen('/dev/null', "rw+");
      $STDERR = fopen('/dev/null', "rw+");
      

加上上面那句,所有的显示的不显示的echo err之类都可以被忽略。也就是说你把
echo 'loop' . $i . "n";这句加上也没有问题
指到dev/null,,如果你不这样,你的stdout会跟你的session有关。。
你的session一关,你的stdout就失效,,echo就报错了。

更优处理办法

C语言实现方式参考

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 


bool my_daemonsize() {
    pid_t pid = fork();
    if (pid < 0) {
        return false;
    } else if ( pid > 0) {
        exit(0);
    }

    umask(0);
    pid_t sid = setsid();

    if (sid < 0) {
        return false;
    }

    if (chdir("/") < 0) {
        return false;
    }
    close(STDIN_FILENO);
    close(STDOUT_FILENO);
    close(STDERR_FILENO);

    open("/dev/null", O_RDONLY);
    open("/dev/null", O_RDWR);
    open("/dev/null", O_RDWR);
    sleep(20);//ps -ef|grep my_daemon |grep -v color
    return true;
}


int main() {
    bool res = my_daemonsize();
    //注意并没有任何输出
    printf("deamon result is%d\n", res);
}

感谢

三金 CFC4N

守护进程详解及创建,daemon()使用

更多要善用手册

当然这只是个例子,实际中还需要考虑目录权限,umask,figchld信号。这些我还没接触。。。
man setpgid
man setsid

你可能感兴趣的:(php,daemon)