编写 PHP 守护进程程序

守护进程(daemon),又称为常驻后台进程。该进程持续在后台运行,处理系统业务。它没有控制终端,不与前台交互。要么手动杀死该进程,要么系统关闭的时候被关闭。通常在小项目当中 PHP 没有此类需求。都是通过编写定时脚本来执行。

今天,我们以完成异步发送短信来编写 PHP 守护进程程序。会讲到编写守护进程程序中会遇到的一些问题。以及这些问题的解决方案。

一、PHP CLI 模式###

PHP CLI 即 命令行模式。这是编写常驻后台程序必须掌握的知识点。关于 PHP CLI 相关的技术细节。可以查看博主之前写的一篇文章《PHP 命令行模式》

我们主要用了 PHP CLI 模式的运行 PHP 脚本的功能。

如:

$ php test.php

二、实例代码

为了避免空洞的理论。我们直接上代码,然后对代码进行抽丝剥茧般分析。再一步一步优化代码,达到我们要求的守护进程级别。

首先,我们要理解异步发送短信的需求涉及的流程。

(1)用户登录/注册等需求短信验证码的位置。点击获取验证码。

(2)服务器收到用户的发送短信请求。将手机号码以及待发送的短信内容放入 Redis 队列。

(3)后台进程持续监听 Redis 队列当中是否有待处理的短信发送。有则发送。无则持续监听。

通过这三步,我们清晰知道。这个异步短信发送的需求会涉及到三个技术点:

(1)队列:存储待发送短信的数据。

(2)把用户短信发送请求写入队列。

(3)从 Redis 队列取出数据进行短信发送。

假设我们的 Redis 队列名称为:sms_list

则写入队列的程序如下:

PushQueue.php 脚本代码如下:

<?php
$redis = new \Redis();
$redis->connect('127.0.0.1', 6379);
$redis->select(1);

$sms = [
    'mobile'  => '14800001234',
    'content' => '您的验证码为:888888。请及时使用,10 分钟后失效。【IT访谈】'
];

$ok = $redis->lPush('sms_list', json_encode($sms, JSON_UNESCAPED_UNICODE));
if ($ok) {
    echo "写入短信队列 sms_list 成功\n";
}

SmsConsume.php 后台消费进程代码如下:

<?php
$redis = new \Redis();
$redis->connect('127.0.0.1', 6379);
$redis->select(1);

$queueKey = 'sms_list';     // 短信队列。
$queueIng = 'sms_list_ing'; // 短处中的队列。

while (true) {
    $content = $redis->bRPopLPush($queueKey, $queueIng, 60);
    if (!empty($content)) {
        $arrCxt = json_decode($content, true);
        /**
         * 调用短信发送接口。
         * 由于是演示代码,此处直接打印输出即可。
         * 真实场景请调用短信发送的接口。
         */
        echo "mobile:{$arrCxt['mobile']}\n";
        echo "content:{$arrCxt['content']}\n\n";
    } else {
        // 暂停 0.1 秒。
        usleep(100000);
    }
}

启动生产端/消费端

(1)启动消费端

$ php SmsConsume.php

启动完成之后,命令终端会一直等待数据写入 Redis 队列。接下来,我们运行生产端往 Redis 队列写入数据。

(2)启动生产端

我们另起一个命令终端执行如下命令:

$ php PushQueue.php

运行成功会输出如下内容:

写入短信队列 sms_list 成功

说明,我们已经成功向 Redis sms_list 队列写入了短信发送的数据。

同时,在我们的消费端命令终端输出了如下内容:

mobile:14800001234
content:您的验证码为:888888。请及时使用,10 分钟后失效。【IT访谈】

问题与缺点:

(1)Redis 读取数据错误

在运行消费端 SmsConsume.php 程序的时候,如果我们的生产端超过 60 秒没有向队列写入数据。消费端在空闲 60 秒之后,会提示类似错误:

...... Uncaught RedisException: read error on connection ......

错误分析:

之所以出现这个错误。是因为在我们的 PHP 配置里面默认限制了一个 socket 连接在 60 秒内没有任何操作就会断开。断开的 socket 连接再去读取数据肯定会报错。此错误依然会出现在 MySQL、Kafka、Memcache 等 socket 连接的系统。

解决方案:

知道了问题所在,剩下的就是更改 PHP 这个默认的配置。

default_socket_timeout = 60

虽然,我们可以直接在 php.ini 文件中修改此值。但是,我们不建议这样做。因为,这个配置不仅会影响 PHP CLI 模式,同时也会影响 PHP CGI 模式(Web 访问)。所以,我们只推荐在代码当中修改。

我们修改 SmsConsume.php 脚本代码之后如下:

<?php

// 防止 Socket 连接空闲超时退出报错。
ini_set('default_socket_timeout', -1);

$redis = new \Redis();
$redis->connect('127.0.0.1', 6379);
$redis->select(1);

$queueKey = 'sms_list';     // 短信队列。
$queueIng = 'sms_list_ing'; // 短处中的队列。

while (true) {
    $content = $redis->bRPopLPush($queueKey, $queueIng, 60);
    if (!empty($content)) {
        $arrCxt = json_decode($content, true);
        /**
         * 调用短信发送接口。
         * 由于是演示代码,此处直接打印输出即可。
         * 真实场景请调用短信发送的接口。
         */
        echo "mobile:{$arrCxt['mobile']}\n";
        echo "content:{$arrCxt['content']}\n\n";
    } else {
        // 暂停 0.1 秒。
        usleep(100000);
    }
}

通过这样修改之后,我们再去运行这个脚本。就会发现不再出现这个错误了。

(2)代码报错进程退出

因为会发生类似 Redis 读取数据错误或其他 PHP 错误。此时,PHP 消费端进程就会终止执行。如果我们把这个消费端程序设置为后端运行的守护进程。这显然是不满足常驻后台运行的目的。

所以,我们需要捕获这些错误。然后写日志或打印到命令行终端。

解决方案:

PHP 提供了 try catch 来解决异常。但是,有时候,PHP 并只是抛出异常,也有可能抛出 Notice、warning 等错误。此时,我们最好的做法是把这些错误转成异常来处理。

在很多成熟的框架都已经将错误转成异常来处理了。所以,我们唯一要做的就是使用 try catch 来捕获异常就行了。

SmsConsume.php 脚本修改之后的代码如下:

<?php

// 防止 Socket 连接空闲超时退出报错。
ini_set('default_socket_timeout', -1);

$redis = new \Redis();
$redis->connect('127.0.0.1', 6379);
$redis->select(1);

$queueKey = 'sms_list';     // 短信队列。
$queueIng = 'sms_list_ing'; // 短处中的队列。

while (true) {
    try {
        $content = $redis->bRPopLPush($queueKey, $queueIng, 60);
        if (!empty($content)) {
            $arrCxt = json_decode($content, true);
            /**
             * 调用短信发送接口。
             * 由于是演示代码,此处直接打印输出即可。
             * 真实场景请调用短信发送的接口。
             */
            echo "mobile:{$arrCxt['mobile']}\n";
            echo "content:{$arrCxt['content']}\n\n";
        } else {
            // 暂停 0.1 秒。
            usleep(100000);
        }
    } catch (\Exception $e) {
        echo "出错了!\n";
        echo "ErrorMsg:" . $e->getMessage() . "\n\n";
    } catch (\Throwable $e) {
        echo "出错了!\n";
        echo "ErrorMsg:" . $e->getMessage() . "\n\n";
    }
}

三、设置消费端为后台运行

我们现在程序已经写好了。现在就需要将程序设置为后台运行。设置为后台运行的方案有很多种。

(1)Linux nohup 命令

关于该命令如何使用,大家可以通过 Google 搜索得到相当全的资料。这里就不用去 Google 搬运了。

(2)Supervisor 管理

这是本博主寒冰推荐的方式。Supervisor 是一款非常优秀的进程管理工具。关于如何使用,可以查看我之前写的一篇文章:CentOS7 安装和使用 Supervisor 工具 。非常详尽怎样使用 Supervisor 这款工具。

四、总结

本篇文章只是一个精简版的守护进程程序。核心的点都已经涉及到。技术的细节方面还需要结合实际的业务进行考量。如果,你在使用本篇文章提到的相关功能时有任何问题,可以留言或者加群(168159147)咨询。谢谢!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容