PHP7 网络编程(二)daemon守护进程

前言php

在一个多任务的计算机操做系统中,守护进程(英语:daemon,/ˈdmən//ˈdmən/)是一种在后台执行的计算机程序。此类程序会被以进程的形式初始化。守护进程程序的名称一般以字母“d”结尾:例如,syslogd就是指管理系统日志的守护进程。html

 

daemon 程序是一直运行的服务端程序,又称为守护进程。一般在系统后台运行,没有控制终端不与前台交互,daemon 程序通常做为系统服务使用。daemon  是长时间运行的进程,一般在系统启动后就运行,在系统关闭时才结束。通常说Daemon程序在后台运行,是由于它没有控制终端,没法和前台的用户交互。daemon程序通常都做为服务程序使用,等待客户端程序与它通讯。咱们也把运行的daemon程序称做守护进程。node

 

一般,守护进程没有任何存在的父进程(即PPID=1),且在UNIX系统进程层级中直接位于init之下。守护进程程序一般经过以下方法使本身成为守护进程:对一个子进程运行fork,而后使其父进程当即终止,使得这个子进程能在init下运行。这种方法一般被称为“脱壳”。mysql

 

系统一般在启动时一同起动守护进程。守护进程为对网络请求,硬件活动等进行响应,或其余经过某些任务对其余应用程序的请求进行回应提供支持。守护进程也可以对硬件进行配置(如在某些Linux系统上的devfsd),运行计划任务(例如cron),以及运行其余任务。每一个进程都有一个父进程,子进程退出,父进程能获得子进程退出的状态。linux

 

守护进程简单地说就是能够脱离终端而在后台运行的进程 . 这在Linux中是很是常见的一种进程 , 好比apache或者mysql等服务启动后 , 就会以守护进程的方式进驻在内存中 。守护程序是在后台运行的应用程序,而不是由用户直接操做。守护进程的例子是Cron和MySQL。 使用PHP守护进程很是简单,而且须要使用PHP 4.1或更高版本编译参数:--enable-pcntlredis

 

假若有个耗时间的任务须要跑在后台 : 将全部mysql中user表中的2000万用户所有导入到redis中作预热缓存 , 那么这个任务估计一时半会是不会结束的 , 这个时候就须要编写一个php脚本以daemon形式运行在系统中 , 结束后自动推出。sql

在Linux中 , 有三种方式实现脚本后台化 :

1 . 在命令后添加一个&符号

好比 php task.php & . 这个方法的缺点在于 若是terminal终端关闭 , 不管是正常关闭仍是非正常关闭 , 这个php进程都会随着终端关闭而关闭 , 其次是代码中若是有echo或者print_r之类的输出文本 , 会被输出到当前的终端窗口中 。shell

2 . 使用nohup命令 

好比 nohup php task.php & . 默认状况下 , 代码中echo或者print_r之类输出的文本会被输出到php代码同级目录的nohup.out文件中 . 若是你用exit命令或者关闭按钮等正常手段关闭终端 , 该进程不会被关闭 , 依然会在后台持续运行 . 可是若是终端遇到异常退出或者终止 , 该php进程也会随即退出 . 本质上 , 也并不是稳定可靠的daemon方案 。apache

3 . 经过 pcntl 与 posix 扩展实现编程

编程中须要注意的地方有:

  • 经过二次 pcntl_fork() 以及 posix_setsid 让主进程脱离终端
  • 经过 pcntl_signal() 忽略或者处理 SIGHUP 信号
  • 多进程程序须要经过二次 pcntl_fork() 或者 pcntl_signal() 忽略 SIGCHLD 信号防止子进程变成 Zombie 进程
  • 经过 umask() 设定文件权限掩码,防止继承文件权限而来的权限影响功能
  • 将运行进程的 STDIN/STDOUT/STDERR 重定向到 /dev/null 或者其余流上

daemon有以下特征:

  • 没有终端
  • 后台运行
  • 父进程 pid 为1

想要查看运行中的守护进程能够经过 ps -ax 或者 ps -ef 查看,其中 -x 表示会列出没有控制终端的进程。

fork 系统调用

       fork 系统调用用于复制一个与父进程几乎彻底相同的进程,新生成的子进程不一样的地方在于与父进程有着不一样的 pid 以及有不一样的内存空间,根据代码逻辑实现,父子进程能够完成同样的工做,也能够不一样。子进程会从父进程中继承好比文件描述符一类的资源。

PHP 中的 pcntl 扩展中实现了 pcntl_fork() 函数,用于在 PHP 中 fork 新的进程。

setsid 系统调用

setsid 系统调用则用于建立一个新的会话并设定进程组 id。这里有几个概念:会话进程组

  在 Linux 中,用户登陆产生一个会话(Session),一个会话中包含一个或者多个进程组,一个进程组又包含多个进程。每一个进程组有一个组长(Session Leader),它的 pid 就是进程组的组 id。进程组长一旦打开一个终端,这一个终端就被称为控制终端。一旦控制终端发生异常(断开、硬件错误等),会发出信号到进程组组长。

  后台运行程序(如 shell 中以&结尾执行指令)在终端关闭以后也会被杀死,就是没有处理好控制终端断开时发出的SIGHUP信号,而SIGHUP信号对于进程的默认行为则是退出进程。

调用 setsid 系统调用以后,会让当前的进程新建一个进程组,若是在当前进程中不打开终端的话,那么这一个进程组就不会存在控制终端,也就不会出现由于关闭终端而杀死进程的问题。

PHP 中的 posix 扩展中实现了 posix_setsid() 函数,用于在 PHP 中设定新的进程组。

二次 fork 的做用

首先,setsid 系统调用不能由进程组组长调用,会返回-1。

二次 fork 操做的样例代码以下:

$pid1 = pcntl_fork(); if ($pid1 > 0) {
// 父进程会获得子进程号,因此这里是父进程执行的逻辑 exit('parent process. 1'."\n"); } else if ($pid1 < 0) { exit("Failed to fork 1\n"); } if (-1 == posix_setsid()) { exit("Failed to setsid\n"); } $pid2 = pcntl_fork(); if ($pid2 > 0) { exit('parent process. 2'."\n"); } else if ($pid2 < 0) { exit("Failed to fork 2\n"); }

pcntl_fork() 函数建立一个子进程,这个子进程仅PID(进程号) 和PPID(父进程号)与其父进程不一样。

返回值

  成功时,在父进程执行线程内返回产生的子进程的PID,在子进程执行线程内返回 0,失败时,在 父进程上下文返回 -1,不会建立子进程,而且会引起一个PHP错误。

假定咱们在终端中执行应用程序,进程为 a,第一次 fork 会生成子进程 b,若是 fork 成功,父进程 a 退出。b 做为孤儿进程,被 init 进程托管。

此时,进程 b 处于进程组 a 中,进程 b 调用 posix_setsid 要求生成新的进程组,调用成功后当前进程组变为 b。

php fork2.php parent process. 1 parent process. 2

此时进程 b 事实上已经脱离任何的控制终端,例程:

cli_set_process_title('process_a'); $pidA = pcntl_fork(); if ($pidA > 0) { exit(0); } else if ($pidA < 0) { exit(1); } cli_set_process_title('process_b'); if (-1 === posix_setsid()) { exit(2); } while(true) { sleep(1); } 

执行程序以后:  

$ php cli-title.php $ ps ax | grep -v grep | grep -E 'process_|PID' PID TTY STAT TIME COMMAND 15725 ? Ss 0:00 process_b

从新打开一个shell窗口,效果同样,都在呢

从 ps 的结果来看,process_b 的 TTY 已经变成了 ,即没有对应的控制终端。

代码走到这里,彷佛已经完成了功能,关闭终端以后 process_b 也没有被杀死,可是为何还要进行第二次 fork 操做呢?

StackOverflow 上的一个回答写的很好:

The second fork(2) is there to ensure that the new process is not a session leader, so it won’t be able to (accidentally) allocate a controlling terminal, since daemons are not supposed to ever have a controlling terminal.

这是为了防止实际的工做的进程主动关联或者意外关联控制终端,再次 fork 以后生成的新进程因为不是进程组组长,是不能申请关联控制终端的。

综上,二次 fork 与 setsid 的做用是生成新的进程组,防止工做进程关联控制终端。 

写一个demo测试下

<?php // 第一次fork系统调用 $pid_A = pcntl_fork(); // 父进程 和 子进程 都会执行下面代码 if ($pid_A < 0) { // 错误处理: 建立子进程失败时返回-1. exit('A fork error '); } else if ($pid_A > 0) { // 父进程会获得子进程号,因此这里是父进程执行的逻辑 exit("A parent process exit \n"); } // B 做为孤儿进程,被 init 进程托管,此时,进程 B 处于进程组 A 中 // 子进程获得的$pid为0, 因此如下是子进程执行的逻辑,受控制终端的影响,控制终端关闭则这里也会退出 // [子进程] 控制终端未关闭前,将当前子进程提高会会话组组长,及进程组的leader // 进程 B 调用 posix_setsid 要求生成新的进程组,调用成功后当前进程组变为 B if (-1 == posix_setsid()) { exit("Failed to setsid\n"); } // 此时进程 B 已经脱离任何的控制终端 // [子进程] 这时候在【进程组B】中,从新fork系统调用(二次fork) $pid_B = pcntl_fork(); if ($pid_B < 0) { exit('B fork error '); } else if ($pid_B > 0) { exit("B parent process exit \n"); } // [新子进程] 这里是新生成的进程组,不受控制终端的影响,写写本身的业务逻辑代码 for ($i = 1; $i <= 100; $i++) { sleep(1); file_put_contents('daemon.log',$i . "--" . date("Y-m-d H:i:s", time()) . "\n",FILE_APPEND); } 

Window 下跑回直接抛出异常

php runtime\daemon.php PHP Fatal error: Uncaught Error: Call to undefined function pcntl_fork() in D:\phpStudy\PHPTutorial\WWW\notes\runtime\daemon.php:13 Stack trace: #0 {main} thrown in D:\phpStudy\PHPTutorial\WWW\notes\runtime\daemon.php on line 13

Linux 下执行,输出结果

php daemon.php
... 97--2018-09-07 03:50:09 98--2018-09-07 03:50:10 99--2018-09-07 03:50:11 100--2018-09-07 03:50:12

因此,如今即便关闭了终端,改脚本任然在后台守护进程运行

总结

以上就是这篇文章的所有内容了,但愿本文的内容对你们的学习或者工做具备必定的参考学习价值,若是有疑问你们能够留言交流,谢谢你们对脚本之家的支持。

参考:

一、https://liaoaoyang.cn/articles/2017/08/22/php-daemon/

二、https://blog.ti-node.com/blog/6343348095782223873

相关文章
相关标签/搜索