一个守护进程通常须要root权限,由于可能要使用特殊端口1-1024及其余权限php
一个守护进程的父进程会被fork以后被杀掉,因此能够说他的父进程是init进程。html
一个守护进程无需交互,也和终端(teriminalsession)无关,因此任何输出,不管是向标准输出仍是错误输出,都须要特殊处理,涉及到的就是stdout和stderrlinux
下面是我用php 编写一个守护进程 demogit
<?php $pid = pcntl_fork(); if ($pid < 0) { echo 'fork error'; exit; } else if ($pid) { //父进程执行到此,返回的为子进程pid,此时须要把父进程杀掉。 echo "fork succ\n"; exit; } else {// 重点是posix_setsid() /* http://linux.die.net/man/2/setsid [setsid详解][1] 主要目的脱离终端控制,自立门户。 建立一个新的会话,并且让这个pid统治这个会话,他既是会话组长,也是进程组长。 并且谁也无法控制这个会话,除了这个pid。固然关机除外。。 这时能够成作pid为这个无终端的会话组长。 注意这个函数须要当前进程不是父进程,或者说不是会话组长。 在这里固然不是,由于父进程已经被kill */ $sid = posix_setsid(); if ($sid < 0) { echo 'setsid error'; exit; } //下面是进行的的daemon test for ($i = 0; $i <= 20; $i++) { echo 'loop' . $i . "\n"; file_put_contents('demo.txt', $i . "--" . date("Y-m-d H:i:s", time()) . "\n", FILE_APPEND); sleep(1); } }
好下面咱们执行操做,输出以下ubuntu
root@tb:/home/tb/linuxing# php php_daemon.php fork succ loop0 root@tb:/home/tb/linuxing# loop1 loop2 loop3 loop4 ... loop20
查看demo.txtsession
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终端Azhong 执行 php php_daemon.php
2.关闭终端A
3.打开新终端B,ps -aux |grep php 发现无此进程函数
若是手慢,本身把握时间或者调整for 次数。。oop
进程从建立它的父进程那里继承了打开的文件描述符。如不关闭,将会浪费系统资源,(这却是小事),形成进程所在的文件系统没法卸下以及引发没法预料的错误。
因此须要关闭这些.net
fclose(STDIN),fclose(STDOUT),fclose(STDERR)
关闭标准输入输出与错误显示。code
<?php $pid = pcntl_fork(); if ($pid < 0) { echo 'fork error'; exit; } else if ($pid) { echo "fork succ\n"; exit; } else { $sid = posix_setsid(); if ($sid < 0) { echo 'setsid error'; exit; } for ($i = 0; $i <= 20; $i++) { // echo 'loop' . $i . "\n"; file_put_contents('demo.txt', $i . "--" . date("Y-m-d H:i:s", time()) . "\n", FILE_APPEND); sleep(1); } }
若是想在关闭当前终端后继续执行
须要关闭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就报错了。
<?php $pid = pcntl_fork(); if ($pid < 0) { echo 'fork error'; exit; } else if ($pid) { echo "fork succ\n"; exit; } else { $sid = posix_setsid(); if ($sid < 0) { echo 'setsid error'; exit; } global $STDOUT, $STDERR; fclose(STDOUT); fclose(STDERR); $STDOUT = fopen('/dev/null', "rw+"); $STDERR = fopen('/dev/null', "rw+"); for ($i = 0; $i <= 20; $i++) { file_put_contents('demo.txt', $i . "--" . date("Y-m-d H:i:s", time()) . "\n", FILE_APPEND); sleep(1); } }
固然这只是个例子,实际中还须要考虑目录权限,umask,figchld信号。这些我还没接触。。。