今天小编给大家分享一下PHP怎么实现守护进程的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。
其实只需要创建子进程并退出父进程,将要处理的工作在子进程中进行就可以实现一个守护进程了。但是仅仅是这么做的话,如果后续任务很复杂,或者引入了一些第三方包,那么可能就会出现奇奇怪怪的问题了。
而在《UNIX环境高级编程》(英语:Advanced Programming in the UNIX Environment,简称APUE)一书中有介绍关于守护进程的编码规范,我们按照规范来实现我们的守护进程就可以避免出现那些奇怪的问题了。而且规范也不复杂,只需要几步就可以了:
创建子进程,退出父进程
子进程创建一个新的会话并成为 session leader
重设文件掩码
改变工作目录
关闭标准输入输出
<?php function daemon() { // [1] 创建子进程 $pid = pcntl_fork(); if ($pid == -1) { die('fork failed'); } // [2] 如果是父进程,则退出 if ($pid > 0) { exit(0); } ///////////////// 以下是子进程 ///////////////// // [3] 创建一个新的会话并成为 session leader if ( ($sid = posix_setsid()) <= 0 ) { die("Set sid failed.\n"); } // [4] 重设文件掩码 umask(0); // [5] 改变工作目录 if (chdir('/') === false) { die("chdir failed.\n"); } // [6] 关闭标准输入输出 fclose(STDIN); fclose(STDOUT); fclose(STDERR); } daemon(); // ... 真正的处理逻辑
上面短短的十几二十行代码就实现了一个守护进程,接下来解释一下有些步骤为什么要这么做。
pcntl_fork()
的返回值有三种情况,上面的代码([1]和[2])已经处理了对应的情况。
调用 posix_setsid()
创建新会话会使得当前进程成为新会话中的“会话首进程”,同时也会使当前进程成为“进程组组长”,并且使得当前进程脱离控制终端。
调用 umask()
重设文件掩码,这里通常是 0。为什么是 0 而不是其他呢,因为子进程从父进程继承来的文件掩码可能会屏蔽某些特定的文件操作权限。比如说引入的第三方库可能需要用特定的权限来创建文件,并且它没有将文件权限作为一个选项参数由你指定,那么就可能会出现失败的情况;而我们传入 0
,会使得从调用了 umask()
之后,守护进程创建的文件权限为 0666
,目录权限为 0777
,均为最高权限。
关于 umask()
后面会展开新的篇幅来说明,感兴趣的可以先自行搜索资料学习。
通过 chdir()
我们将工作目录设置为根目录 /
,主要是因为守护进程是长时间运行的,通常只有系统关闭/重启才会退出。假如从父进程继承来的工作目录是个挂载的文件系统,如果不改变工作目录,那么将会导致这个挂载的文件系统一直没法卸载。
当然也不一定要将工作目录切换到根目录,你也可以根据实际情况切换到特定的目录。
因为守护进程是脱离终端控制的,所以是没有标准输入输出交互的,我们将其关闭即可。
二次 fork
你可能在一些资料中看到有人推荐你在 [3] 创建一个新的会话并成为 session leader 之后再次进行 fork
。这一步骤是在基于 System V
的系统中,可以保证你的守护进程不是“会话首进程”,可以阻止其重新申请获取一个控制终端。
关闭不必要的文件描述符
按照编码规范,实际还有一步是关闭不必要的文件描述符。但我们为了简单起见,上面的代码在进程启动之后先创建守护进程再执行其他操作,因此这里只打开了三个文件描述符: 0
、1
和 2
(即标准输入、标准输出、标准错误)。
因为上面的代码将标准输入输出关闭了,也就是说如果你在 daemon()
之后有诸如 echo "Hello world";
之类的输出,那么你的程序将会出错然后退出,并且你将看不到任何错误信息(因为标准错误也被关闭了)。
解决方案有两种,一种是用 file_put_contents
代替 echo
,但是这样并不优雅,而且万一引入的第三方包中写了 echo
或者是 file_put_contents(STDOUT, ...)
,那你的程序也会“莫名其妙”就挂了,会让你排查半天到底是哪里出了问题。
因此我们还可以在第[6]之后加入:
// [7] 重定向输入输出 global $stdin, $stdout, $stderr; $stdin = fopen('/dev/null', 'r'); $stdout = fopen('/dev/null', 'wb'); // 你也可以将标准输出重定向到指定的文件,相当于是日志 $stderr = fopen('/dev/null', 'wb'); // 同上
以上就是“PHP怎么实现守护进程”这篇文章的所有内容,感谢各位的阅读!相信大家阅读完这篇文章都有很大的收获,小编每天都会为大家更新不同的知识,如果还想学习更多的知识,请关注亿速云行业资讯频道。
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。