From ad0cdee631df393a9bfb30ca2785d65dd31742c3 Mon Sep 17 00:00:00 2001 From: elarity <42677833@qq.com> Date: Tue, 24 Apr 2018 22:17:20 +0800 Subject: [PATCH] first --- ...43\347\241\256\345\201\232\346\263\225.md" | 35 +++++++++++++++++++ 1 file changed, 35 insertions(+) create mode 100644 "php\350\277\233\347\250\213daemon\345\214\226\347\232\204\346\255\243\347\241\256\345\201\232\346\263\225.md" diff --git "a/php\350\277\233\347\250\213daemon\345\214\226\347\232\204\346\255\243\347\241\256\345\201\232\346\263\225.md" "b/php\350\277\233\347\250\213daemon\345\214\226\347\232\204\346\255\243\347\241\256\345\201\232\346\263\225.md" new file mode 100644 index 0000000..45f7ca3 --- /dev/null +++ "b/php\350\277\233\347\250\213daemon\345\214\226\347\232\204\346\255\243\347\241\256\345\201\232\346\263\225.md" @@ -0,0 +1,35 @@ +###### daemon 音标 : ['di:mən] , 中文含义为守护神或精灵的意思 . 其实它还有个意思 : 守护进程 . +###### 守护进程简单地说就是可以脱离终端而在后台运行的进程 . 这在Linux中是非常常见的一种进程 , 比如apache或者mysql等服务启动后 , 就会以守护进程的方式进驻在内存中 . +###### 以PHP为例 , 假如我有个耗时间的任务需要跑在后台 : 将所有mysql中user表中的2000万用户全部导入到redis中做预热缓存 , 那么这个任务估计一时半会是不会结束的 , 这个时候就需要编写一个php脚本以daemon形式运行在系统中 , 结束后自动推出 . +###### 在Linux中 , 大概有三种方式实现脚本后台化 : +###### 1 . 在命令后添加一个&符号 , 比如 php task.php & . 这个方法的缺点在于 如果terminal终端关闭 , 无论是正常关闭还是非正常关闭 , 这个php进程都会随着终端关闭而关闭 , 其次是代码中如果有echo或者print_r之类的输出文本 , 会被输出到当前的终端窗口中 . +###### 2 . 使用nohup命令 , 比如 nohup php task.php & . 默认情况下 , 代码中echo或者print_r之类输出的文本会被输出到php代码同级目录的nohup.out文件中 . 如果你用exit命令或者关闭按钮等正常手段关闭终端 , 该进程不会被关闭 , 依然会在后台持续运行 . 但是如果终端遇到异常退出或者终止 , 该php进程也会随即退出 . 本质上 , 也并非稳定可靠的daemon方案 . +###### 3 . 使用fork和setsid , 我暂且称之为 : *nix解决方案 . 具体看下代码 : +```php + 0 ) { + exit( ' parent process. ' ); + } + // 将当前子进程提升会会话组组长 这是至关重要的一步 + if ( ! posxi_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 ); + } +?> +```