微麦PHP 2019-11-03
一、 进程守护使用场景。
后端经常会有类似这样的场景,某个脚本,需要不断的重复运行,这个时候,最好有一个守护程序,帮助我们不断地自动地拉起这些脚本进程,让它自动地重复运行。
在 Linux/Unix 系统下,supervisor 就是使用 python 开发的一个优秀的进程管理工具,本文尝试使用 php 来实现类似的进程管理工具。
二、swoole 的进程管理模块。
php 的 swoole 扩展有一个进程管理模块,官方文档见:swoole 进程管理模块
参考 supervisor 的实现方式,被守护的进程是作为 supervisor 的子进程来启动的,supervisor 通过监听子进程的信号,可实现对子进程的自动重启等功能。而 swoole 的进程管理模块就提供了进程间通信的功能,可以实现对子进程的自动重启功能。
三、第一个进程守护程序。
要实现对子程序的守护,需要做到两点:
swoole 进程管理模块提供了一个 bool Process->exec(string $execfile, array $args) 方法,让子进程蜕变成另一个系统调用程序,同时还能保证父进程与当前进程仍然是父子进程关系。
再通过 array Process::wait(bool $blocking = true) 方法,来等待子进程的退出信号。
下面是使用 swoole 启动子进程,并回收子进程资源的示例代码:
<?php use Swoole\Process; $php = "/usr/bin/env php"; $script = dirname(__DIR__) . "/task.php"; $command = "{$php} {$script}"; $process = new Process(function (Process $worker) use ($command) { $worker->exec('/bin/sh', ['-c', $command]); }); $pid = $process->start(); printf("启动子进程 {$pid}\n"); while ($ret = Process::wait()) { $pid = intval($ret["pid"] ?? 0); printf("子进程 {$pid} 结束\n"); }
代码解析:
$command 变量表示需要子进程脚本,通过 exec() 方法来启动成一个子进程的方式运行,再通过 Process::wait() 访求来等待 $command 这个子进程脚本结束,并回收进程资源。
那么,只要在收到子进程的结束信号后,再起一个相同的子进程脚本,即可实现对子进程的守护了。于是,第一个守护子进程的程序实现代码:
<?php use Swoole\Process; $php = "/usr/bin/env php"; $script = dirname(__DIR__) . "/task.php"; $command = "{$php} {$script}"; do { $process = new Process(function (Process $worker) use ($command) { $worker->exec('/bin/sh', ['-c', $command]); }); $pid = $process->start(); printf("启动子进程 {$pid}\n"); } while (Process::wait());
代码解析:
这段代码只将启动子进程的逻辑加到一个死循环中,好让这个子进程脚本能够不断的重启。
四、封装成类
为了方便重用这段代码,可以将这段代码封装成一个简单的类:
<?php namespace App; use Swoole\Process; class Daemon { /** @var string */ private $command; public function __construct(string $command) { $this->command = $command; } public function run() { do { $process = new Process(function (Process $worker) { $worker->exec('/bin/sh', ['-c', $this->command]); }); $pid = $process->start(); } while (Process::wait()); } }
那么,这个 Daemon 类的使用方式如下:
<?php use App\Daemon; $php = "/usr/bin/env php"; $script = dirname(__DIR__) . "/task.php"; $command = "{$php} {$script}"; $daemon = new Daemon($command); $daemon->run();
这个简单 Daemon 类虽然能实现对单个脚本进行重启守护,但是,如果我们有许多个脚本同时需要守护的,这个 Daemon 类显然是不能够满足需求的。
下一篇文章 使用 swoole 实现进程的守护(二)将尝试扩展这个 Daemon 类。
<?php. if (!empty($_POST)) {. $data1 = $_POST["data1"];$data2 = $_POST["data2"];$fuhao = $_POST["fuh