本文已参与「新人创作礼」活动,一起开启掘金创作之路。
PHP 进程 daemon 化的做法
初步了解
daemon 音标 : [‘di:mən] , 中文含义为守护神或精灵的意思 。 其实它还有个意思 : 守护进程
守护进程简单地说就是可以脱离终端而在后台运行的进程 。 这在 Linux 中是非常常见的一种进程, 比如 Apache 或者 Mysql 等服务启动后,就会以守护进程的方式进驻在内存中 。
以 PHP 为例 , 假如我有个耗时间的任务需要跑在后台 : 将所有 Mysql 中 user 表中的 2000万 用户全部导入到 Redis 中做缓存预热, 那么这个任务估计一时半会是不会结束的 , 这个时候就需要编写一个 php 脚本以 daemon 形式运行在系统中,结束后自动退出。
实际操作
在 Linux 中 , 大概有三种方式实现脚本后台化 :
- 在命令后添加一个 & 符号 , 比如 php task.php &。 这个方法的缺点在于如果 terminal 终端关闭,无论是正常关闭还是非正常关闭,这个 php 进程都会随着终端关闭而关闭,其次是代码中如果有 echo 或者 print_r 之类的输出文本,会被输出到当前的终端窗口中 。
- 使用 nohup 命令 , 比如 nohup php task.php & 。 默认情况下,代码中 echo 或者 print_r 之类输出的文本会被输出到 php 代码同级目录的 nohup.out 文件中。如果你用 exit 命令或者关闭按钮等正常手段关闭终端,该进程不会被关闭,依然会在后台持续运行。但是如果终端遇到异常退出或者终止,该 php 进程也会随即退出。本质上,也并非稳定可靠的 daemon 方案。
- 使用 fork 和 setsid,暂且称之为 : *nix 解决方案。具体看下代码 :
<?php
// 一次fork
$pid = pcntl_fork();
if ( $pid < 0 ) {
exit( ' fork error. ' );
} else if( $pid > 0 ) {
exit( ' parent process. ' );
}
// 将当前子进程提升会会话组组长 这是至关重要的一步
if ( ! posix_setsid() ) {
exit( ' setsid error. ' );
}
// 二次fork
$pid = pcntl_fork();
if( $pid < 0 ){
exit( ' fork error. ' );
} else if( $pid > 0 ) {
exit( ' parent process. ' );
}
// 真正的逻辑代码们 下面仅仅写个循环以示例
for( $i = 1 ; $i <= 100 ; $i++ ){
sleep( 1 );
file_put_contents( 'daemon.log', $i, FILE_APPEND );
}
?>
另外还有一种运行脚本的方式也值得一提:利用 screen / tmux 等软件,将脚本运行在可以在一个虚拟终端之上。
- END -