消息队列 think-queue tp5.0

一 介绍

think-queue是tp框架消息队列插件,依赖框架框架核心版本,所以下载时需要选择对应框架核心的版本。

比如tp5.1用2.0的都可以,5.0的用1.1.6。其余版本参考composer。

composer地址:topthink/think-queue - Packagist

不同版本中项目结构不同,一般会说明插件的使用方法,比如配置文件位置。可以在项目中查找"queue.php"通过md文件查看说明,或者通过源码分析获取其存贮位置。

由于本地使用tp5.0版本,只有内容都仅适用tp5.0。

消息队列用于解决并发问题。比如处理微信支付回执......目前退款和支付总是同时给出同一单多个回执,连触发器都没拦截住,只能考虑消息队列。

二 安装&配置

bash 复制代码
composer require topthink/think-queue v1.1.6

配置文件位于 "application/extra/queue.php"。

php 复制代码
#application/extra/queue.php
return [
    //驱动引擎
    'connector' => 'Redis',
    //失效时间
    'expire' => 60,
    //默认队列名
    'default' => 'hirorder',
    //ip地址
    'host' => '127.0.0.1',
    //端口
    'port' => 6379,
    //密码
    'password' => '',
    //选择数据库
    'select' => 0,
    //超时时间
    'timeout' => 0,
    //是否持久化
    'persistent' => false,
];

各个驱动的具体可用配置项在"think\queue\connector"目录下。各个驱动类里的`options`属性是配置数据,写在上面的"queue"配置里即可覆盖。

不同驱动配置数据不同,可以在"think\queue\connector"中查看"options"属性。

三 原理

用redis模拟消息队列的原理很简单。

推送消息到队列中,即redis的list中。

若队列不存在会在redis中新建对应list,格式如queue:队列名。

安装插件后,可使用queue:work和queue:listen等命令。

开启两者都需要设置参数,比如work的"queue"为队列名、work的"daemon"为是否开启守护线程。

对应命令在"think-queue\src\queue\command"可查。

3.1 推送信息

php 复制代码
use think\Queue;

// 1.当前任务将由哪个类来负责处理。
// 当轮到该任务时,系统将生成一个该类的实例,并调用其 fire 方法
$jobHandlerClassName = 'app\jobs\JobTest';

// 2.当前任务归属的队列名称,如果为新队列,会自动创建
$jobQueueName = "helloJobQueue";

// 3.当前任务所需的业务数据, 不能为 resource 类型,其他类型最终将转化为json形式的字符串
// (jobData 为对象时,存储其public属性的键值对 )
$jobData = ['ts' => time(), 'bizId' => uniqid(), 'a' => 1];

// 4.将该任务推送到消息队列,等待对应的消费者去执行
$isPushed = Queue::push($jobHandlerClassName, $jobData, $jobQueueName);

// database 驱动时,返回值为 1|false;redis 驱动时,返回值为 随机字符串|false
if ($isPushed !== false) {
   echo date('Y-m-d H:i:s') . " a new Hello Job is Pushed to the MQ" . "<br>";
} else {
  echo 'Oops, something went wrong.';
}

$jobHandlerClassName为处理数据的类,需要自己定义。

php 复制代码
#application\jobs\JobTest
namespace app\jobs;
use think\queue\Job;
use app\common\model\JobsTest as JobsTestModel;
class JobTest {
    /**
     * fire方法是消息队列默认调用的方法
     * @param Job            $job      当前的任务对象
     * @param array|mixed    $data     发布任务时自定义的数据
     */
    public function fire(Job $job, $data) {
        // 此处做一些 check,提前判断是否需要执行
        $isJobStillNeedToBeDone = $this->checkDatabaseToSeeIfJobNeedToBeDone($data);
        if (!$isJobStillNeedToBeDone) {
            $job->delete();
            return;
        }
        // 执行逻辑处理(即:你需要该消息队列做什么)
        $isJobDone = $this->doHelloJob($data);
        if ($isJobDone) {
            // 如果任务执行成功,记得删除任务
            $job->delete();
        } else {
            // 通过这个方法可以检查这个任务已经重试了几次了
            if ($job->attempts() > 3) {
                $job->delete();
                // 也可以重新发布这个任务
                //$job->release(2); // $delay为延迟时间,表示该任务延迟2秒后再执行
            }
        }
    }

    /**
     * 有些消息在到达消费者时,可能已经不再需要执行了
     * @param $data 发布任务时自定义的数据
     * @return bool 任务执行的结果
     */
    private function checkDatabaseToSeeIfJobNeedToBeDone($data) {
        return true;
    }

    /**
     * 根据消息中的数据进行实际的业务处理...
     * @param $data
     * @return bool
     */
    private function doHelloJob($data) {
        // TODO 该处为实际业务逻辑,即:对消息中的数据进行处理
        // $model = new JobsTestModel();
        // $inData = [
        //     'uniqId' => $data['uniqId'],
        //     'time' => $data['ts'],
        //     'content' => '队列成功的插入数据'
        // ];
        // $res = $model->save($inData);
        // if (! $res) {
        //     return false;
        // }
        // return true;
        var_dump("time:" . time() . " doHelloJob");
    }
}

源码如下

php 复制代码
#vendor\topthink\think-queue\src
class Queue
{
    /** @var Connector */
    protected static $connector;

    private static function buildConnector()
    {
        $options = Config::get('queue');
        $type    = !empty($options['connector']) ? $options['connector'] : 'Sync';

        if (!isset(self::$connector)) {

            $class = false !== strpos($type, '\\') ? $type : '\\think\\queue\\connector\\' . Str::studly($type);

            self::$connector = new $class($options);
        }
        return self::$connector;
    }

    public static function __callStatic($name, $arguments)
    {
        return call_user_func_array([self::buildConnector(), $name], $arguments);
    }
}
php 复制代码
#vendor\topthink\think-queue\src\queue\connector
class Redis extends Connector
{
protected $options = [
        'expire'     => 60,
        'default'    => 'default',
        'host'       => '127.0.0.1',
        'port'       => 6379,
        'password'   => '',
        'select'     => 0,
        'timeout'    => 0,
        'persistent' => false
    ];
public function __construct($options)
    {
        if (!extension_loaded('redis')) {
            throw new Exception('redis扩展未安装');
        }
        if (!empty($options)) {
            $this->options = array_merge($this->options, $options);
        }

        $func        = $this->options['persistent'] ? 'pconnect' : 'connect';
        $this->redis = new \Redis;
        $this->redis->$func($this->options['host'], $this->options['port'], $this->options['timeout']);

        if ('' != $this->options['password']) {
            $this->redis->auth($this->options['password']);
        }

        if (0 != $this->options['select']) {
            $this->redis->select($this->options['select']);
        }
    }
public function push($job, $data = '', $queue = null)
    {
        return $this->pushRaw($this->createPayload($job, $data), $queue);
    }
 public function pushRaw($payload, $queue = null)
    {
        $this->redis->rPush($this->getQueue($queue), $payload);

        return json_decode($payload, true)['id'];
    }
protected function createPayload($job, $data = '', $queue = null)
    {
        $payload = $this->setMeta(
            parent::createPayload($job, $data), 'id', $this->getRandomId()
        );

        return $this->setMeta($payload, 'attempts', 1);
    }
protected function getQueue($queue)
    {
        return 'queues:' . ($queue ?: $this->options['default']);
    }
}
php 复制代码
#vendor\topthink\think-queue\src\queue
abstract class Connector
{
protected function setMeta($payload, $key, $value)
    {
        $payload       = json_decode($payload, true);
        $payload[$key] = $value;
        $payload       = json_encode($payload);

        if (JSON_ERROR_NONE !== json_last_error()) {
            throw new InvalidArgumentException('Unable to create payload: ' . json_last_error_msg());
        }

        return $payload;
    }
protected function createPayload($job, $data = '', $queue = null)
    {
        if (is_object($job)) {
            $payload = json_encode([
                'job'  => 'think\queue\CallQueuedHandler@call',
                'data' => [
                    'commandName' => get_class($job),
                    'command'     => serialize(clone $job),
                ],
            ]);
        } else {
            $payload = json_encode($this->createPlainPayload($job, $data));
        }

        if (JSON_ERROR_NONE !== json_last_error()) {
            throw new InvalidArgumentException('Unable to create payload: ' . json_last_error_msg());
        }

        return $payload;
    }
protected function createPlainPayload($job, $data)
    {
        return ['job' => $job, 'data' => $data];
    }
}

think\Queue::push()执行think\queue\connector\Redis::__construct()调用think\queue\connector\Redis::push()。

push()中将通过父类方法think\queue\Connector::createPayload()和think\queue\Connector::setMeta() 创建的参数传入 think\queue\connector\Redis::pushRaw()中。

think\queue\Connector::createPayload()中逻辑$job可以为对象,其为对象和非对象构造json的数据格式不相同。

pushRaw()使用think\queue\connector\Redis::getQueue()设置list名字。其中使用redis的rpush命令推送数据,该命令若list不存在则新建。

getQueue()中若未设置队列名称则会使用配置中的默认名称,若未设置应会报错。

3.2 处理信息

信息处理使用其自带的命令。命令文件在"think-queue\src\queue\command"中。

一般使用"queue:work"或"queue:listen"。

以queue:work为例。

线程执行

bash 复制代码
php think queue:work --queue list名 --daemon

线程重启

php 复制代码
php think queue:restart

源码比较多,涉及vendor\topthink\think-queue\src\queue\command\Work.php、vendor\topthink\think-queue\src\queue\command\Restart.php、vendor\topthink\think-queue\src\queue\Worker.php等。

command\Work.php中若使用守护线程则会使用死循环,死循环中执行\queue\Worker.php中的pop()函数。

think\queue\Worker::pop()中会调用think\queue\Worker::process()和think\queue\Worker::sleep()。

hink\queue\Worker::process()中通过think\queue\connectorResit::pop()(使用redis驱动)获取实例对象为$job。

其获取实例的过程,实际是解析存储的json数据,通过其中job参数实例化类,使用函数think\queue\Job::resolveAndFire()和think\queue\Job::resolve()。

hink\queue\Worker::process()中获取对象job后调用job->fire(),即用户定义的处理数据类中的fire()函数。

即总过程就像死循环中执行逻辑然后睡几秒后继续执行。若逻辑代码改变,需要线程重启。

若"queue:work"命令行未设置队列名也是使用默认队列名。

想一个命令执行全部队列,估计得另外写代码,应该也不难使用redis的keys匹配出数据,稍微修改下command\Work.php中的代码。

php 复制代码
public function execute(Input $input, Output $output)
    {
    $queue = $input->getOption('queue');//替换掉

    $delay = $input->getOption('delay');

    $memory = $input->getOption('memory');

    if ($input->getOption('daemon')) {
            Hook::listen('worker_daemon_start', $queue);
            $this->daemon(
                $queue, $delay, $memory,
                $input->getOption('sleep'), $input->getOption('tries')
            );
    } else {
            $response = $this->worker->pop($queue, $delay, $input->getOption('sleep'),         $input->getOption('tries'));
            $this->output($response);
    }
}

3.3 进程管理

两者不同参考:

解析think-queue(围绕redis做分析)-ThinkPHP-PHP中文网

https://segmentfault.com/a/1190000020003672

总结:work 模式的性能会比listen模式高,但需要手动执行。listen不用手动执行。

感觉给work命令做个systemd好些,查过一些线程管理的linux应用。

其他方法参考:

thinkphp队列think-queue的使用以及通过supervisor实现常驻进程 - 简书

代码后台运行,不止nohup,还有Python Supervisor!

supervisor · PyPI

https://www.cnblogs.com/mklblog/p/18070785

四 使用

php 复制代码
#application\index\command\jobtest.php
namespace app\index\command;

use think\console\Command;
use think\console\Input;
use think\console\Output;
use think\Queue;

class jobtest extends Command {
    protected function configure() {
        $this->setName('jobtest')->setDescription('job queue test');
    }
    protected function execute(Input $input, Output $output) {
        // 1.当前任务将由哪个类来负责处理。
        // 当轮到该任务时,系统将生成一个该类的实例,并调用其 fire 方法
        $jobHandlerClassName = 'app\jobs\JobTest';

        // 2.当前任务归属的队列名称,如果为新队列,会自动创建
        $jobQueueName = "helloJobQueue";

        // 3.当前任务所需的业务数据, 不能为 resource 类型,其他类型最终将转化为json形式的字符串
        // (jobData 为对象时,存储其public属性的键值对 )
        $jobData = ['ts' => time(), 'bizId' => uniqid(), 'a' => 1];

        // 4.将该任务推送到消息队列,等待对应的消费者去执行
        $isPushed = Queue::push($jobHandlerClassName, $jobData, $jobQueueName);

        // database 驱动时,返回值为 1|false;redis 驱动时,返回值为 随机字符串|false
        if ($isPushed !== false) {
            echo date('Y-m-d H:i:s') . " a new Hello Job is Pushed to the MQ" . "<br>";
        } else {
            echo 'Oops, something went wrong.';
        }
    }
}
php 复制代码
#application\common.php
return [
    // 'app\index\command\test1',
    'app\index\command\jobtest',
];
php 复制代码
#application\jobs\JobTest
namespace app\jobs;
use think\queue\Job;
use app\common\model\JobsTest as JobsTestModel;
class JobTest {
    /**
     * fire方法是消息队列默认调用的方法
     * @param Job            $job      当前的任务对象
     * @param array|mixed    $data     发布任务时自定义的数据
     */
    public function fire(Job $job, $data) {
        // 此处做一些 check,提前判断是否需要执行
        $isJobStillNeedToBeDone = $this->checkDatabaseToSeeIfJobNeedToBeDone($data);
        if (!$isJobStillNeedToBeDone) {
            $job->delete();
            return;
        }
        // 执行逻辑处理(即:你需要该消息队列做什么)
        $isJobDone = $this->doHelloJob($data);
        if ($isJobDone) {
            // 如果任务执行成功,记得删除任务
            $job->delete();
        } else {
            // 通过这个方法可以检查这个任务已经重试了几次了
            if ($job->attempts() > 3) {
                $job->delete();
                // 也可以重新发布这个任务
                //$job->release(2); // $delay为延迟时间,表示该任务延迟2秒后再执行
            }
        }
    }

    /**
     * 有些消息在到达消费者时,可能已经不再需要执行了
     * @param $data 发布任务时自定义的数据
     * @return bool 任务执行的结果
     */
    private function checkDatabaseToSeeIfJobNeedToBeDone($data) {
        return true;
    }

    /**
     * 根据消息中的数据进行实际的业务处理...
     * @param $data
     * @return bool
     */
    private function doHelloJob($data) {
        // TODO 该处为实际业务逻辑,即:对消息中的数据进行处理
        // $model = new JobsTestModel();
        // $inData = [
        //     'uniqId' => $data['uniqId'],
        //     'time' => $data['ts'],
        //     'content' => '队列成功的插入数据'
        // ];
        // $res = $model->save($inData);
        // if (! $res) {
        //     return false;
        // }
        // return true;
          sleep(2);
        var_dump("time:" . date("Y-m-d H:i:s") . " doHelloJob");
    }
}
php 复制代码
php think queue:work --queue helloJobQueue --daemon

php think jobtest

压入数据后redis结果

php 复制代码
> keys queues:*
1) "queues:helloJobQueue"
2) "queues:helloJobQueue:reserved"

测试结果

php 复制代码
string(35) "time:2024-09-03 11:31:20 doHelloJob"
Processed: app\jobs\JobTest
string(35) "time:2024-09-03 11:31:22 doHelloJob"
Processed: app\jobs\JobTest

改变逻辑

php 复制代码
private function doHelloJob($data) {
         sleep(2);
        var_dump("time:" . time() . "(0_0)" . " doHelloJob");
    }

若不重启代码无效,测试结果

php 复制代码
string(35) "time:2024-09-03 11:32:23 doHelloJob"
Processed: app\jobs\JobTest
string(35) "time:2024-09-03 11:32:37 doHelloJob"
Processed: app\jobs\JobTest
php 复制代码
php think queue:restart

queue:restar 后线程终止,需手动启动。在windows中测试......

实际开发应根据开发环境再做调整。

压入数据的程序不限制调用位置,甚至可以作为Hook。

相关推荐
Erik_LinX22 分钟前
day1-->day7| 机器学习(吴恩达)学习笔记
笔记·学习·机器学习
索然无味io37 分钟前
组件框架漏洞
前端·笔记·学习·安全·web安全·网络安全·前端框架
珊瑚里的鱼1 小时前
【单链表算法实战】解锁数据结构核心谜题——环形链表
数据结构·学习·程序人生·算法·leetcode·链表·visual studio
林涧泣1 小时前
图的矩阵表示
学习·线性代数·矩阵
chimchim661 小时前
【starrocks学习】之catalog
学习
梦云澜2 小时前
论文阅读(二):理解概率图模型的两个要点:关于推理和学习的知识
论文阅读·深度学习·学习
Ronin-Lotus2 小时前
上位机知识篇---CMake
c语言·c++·笔记·学习·跨平台·编译·cmake
lxl13073 小时前
学习数据结构(2)空间复杂度+顺序表
数据结构·学习
Мартин.3 小时前
[Meachines] [Easy] Bashed PHP Bash+Python计划任务权限提升
python·php·bash
小机学AI大模型3 小时前
关于使用PHP时WordPress排错——“这意味着您在wp-config.php文件中指定的用户名和密码信息不正确”的解决办法
开发语言·php