文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

PHP怎么实现平滑关闭和重启

2023-06-30 14:37

关注

本篇内容介绍了“PHP怎么实现平滑关闭和重启”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

原理

要实现平滑关闭/重启不难,这里先讲解两个知识点:

阻塞信号

当我们的程序正在处理一个任务的时候,你肯定不希望它中途被终止,比如说你在执行一个数据库事务,肯定不希望事务还没被提交进程就被终止了。

<?phpecho "开始执行事务" . PHP_EOL;// 模拟一些耗时的操作$finish_time = time() + 5;while (time() < $finish_time) {}echo "事务执行完毕" . PHP_EOL;

上面这段代码,如果你在第二个 echo 之前用 kill 命令去杀死这个进程,那么第二个 echo 就不会被执行了。那能不能做到在事务过程中暂时先忽略 kill 信号呢?

能。我们可以使用 pcntl_sigprocmask() 来阻塞信号,让事务完成之后再响应 kill 信号。

<?php// 阻塞信号$sig_set = array(SIGINT, SIGTERM); // 要阻塞的信号集合pcntl_sigprocmask(SIG_BLOCK, $sig_set); // SIG_BLOCK: 把信号加入到当前阻塞信号中echo date("[Y-m-d H:i:s]") . " 开始执行事务" . PHP_EOL;$finish_time = time() + 5;while (time() < $finish_time) {}echo date("[Y-m-d H:i:s]") . "事务执行完毕" . PHP_EOL;pcntl_sigprocmask(SIG_UNBLOCK, $sig_set); // SIG_UNBLOCK: 从当前阻塞信号中移出信号

同样的,在第二个 echo 之前按下 Ctrl + C 或者用 kill 命令去杀这个进程,你会发现第二个 echo 正常执行了,并且两条输出的时间间隔是 5 秒。

我们的常驻进程通常是在一个 while(true) 循环中去执行重复的任务,如果这么写的话:

<?phpwhile (true) {    pcntl_sigprocmask(SIG_BLOCK, $sig_set);    // ...    pcntl_sigprocmask(SIG_UNBLOCK, $sig_set);}

我们是可以保证一个事务不会被打断,但是我们的程序还不知道是不是已经接收到信号了,并且把阻塞信号移除之后进程立刻就退出了,没办法去做一些收尾工作(比如关闭文件)。

处理信号

为了解决上面提到的问题,我们需要在信号发生的时候去做收尾工作,然后再退出进程。

pcntl 扩展提供了一些信号相关的函数,我们可以使用 pcntl_signal() 和 pcntl_signal_dispatch() 来注册信号处理器和分发信号。

<?php$sig_handler = function ($signo) {    echo "收到信号 {$signo}" . PHP_EOL;};pcntl_signal(SIGINT, $sig_handler); // 给 SIGINT 信号注册一个处理器// 模拟耗时操作echo "开始执行事务" . PHP_EOL;$finish_time = time() + 5;while(true) {    if (time() > $finish_time) {        echo "事务执行完毕" . PHP_EOL;        break;    }}pcntl_signal_dispatch(); // 分发信号

执行上面这段代码并在 5 秒内按下 Ctrl + C,你会看到 sig_handler 被执行了;而如果不按下 Ctrl + C,那么 sig_handler 就不会被执行。

到这里你应该已经理解了 pcntl_signal() 和 pcntl_signal_dispatch() 的用法了,把它放到到刚刚的代码试试

<?php$sig_handler = function ($signo) {    echo "收到信号 {$signo}" . PHP_EOL;};$sig_set = array(SIGINT, SIGTERM);foreach ($sig_set as $sig) {    pcntl_signal($sig, $sig_handler); // 注册多个信号}// [1]while (true) {    // [2-1]    pcntl_sigprocmask(SIG_BLOCK, $sig_set);    // [2-2]    // ...    // [2-3]    pcntl_sigprocmask(SIG_UNBLOCK, $sig_set);    // [2-4]}// [3]

pcntl_signal_dispatch() 该放哪里呢?是 [1] [2] 还是 [3]?先动手试一下

然后你会发现,只有放在 [2] 才能让信号处理器执行。同时这个实验也告诉我们 pcntl_signal_dispatch() 要在信号发生后才会使处理器执行:放在 [1] 时,除非你手速足够快,不然在你按下 Ctrl + C 或者是 kill 之前就已经执行过了;而放在 [3] 它就永远没机会执行。

至于放在 [2] 的哪个位置,我建议是放在 [2-4],因为这个时候已经处理完任务了。

拼起来

到这里你已经了解平滑关闭/重启的原理了,我们把上面的半成品代码(因为在收到信号后可能还会进入下一层循环)整理一下:

<?php$running = true;$sig_handler = function ($signo) use (&$running) {    echo "收到信号 {$signo}" . PHP_EOL;    // 做收尾工作    $running = false;};$sig_set = array(SIGINT, SIGTERM, SIGUSR2 );foreach ($sig_set as $sig) {    pcntl_signal($sig, $sig_handler); // 注册多个信号}while ($running) {    pcntl_sigprocmask(SIG_BLOCK, $sig_set);    // ... 业务逻辑    pcntl_sigprocmask(SIG_UNBLOCK, $sig_set);    pcntl_signal_dispatch();}

我们就得到了一个可以平滑程序的常驻进程框架,你也可以把它封装成一个类。

“PHP怎么实现平滑关闭和重启”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注编程网网站,小编将为大家输出更多高质量的实用文章!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯