何为守护进程

守护进程是在后台运行不受终端控制的进程(如输入、输出等)

守护进程脱离终端的主要原因有两点:

用来启动守护进程的终端在启动守护进程之后,需要执行其他任务。

(如其他用户登录该终端后,以前的守护进程的错误信息不应出现)由终端上的一些键所产生的信号(如中断信号),不应对以前从该终端上启动的任何守护进程造成影响

创建守护进程的过程

调用fork创建子进程。父进程终止,让子进程在后台继续执行。

子进程调用setsid产生新会话期并失去控制终端调用setsid()使子进程进程成为新会话组长和新的进程组长,同时失去控制终端。

忽略SIGHUP信号。会话组长进程终止会向其他进程发该信号,造成其他进程终止。

调用fork再创建子进程。子进程终止,子子进程继续执行,由于子子进程不再是会话组长,从而禁止进程重新打开控制终端。

改变当前工作目录为根目录。一般将工作目录改变到根目录,这样进程的启动目录也可以被卸掉。

关闭打开的文件描述符,打开一个空设备,并复制到标准输出和标准错误上。 避免调用的一些库函数依然向屏幕输出信息。

重设文件创建掩码清除从父进程那里继承来的文件创建掩码,设为0。

用openlog函数建立与syslogd的连接。

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