第一次提交
This commit is contained in:
parent
0730c5f3a5
commit
179a39675d
|
@ -0,0 +1,161 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Redis实现延迟消息队列
|
||||||
|
* Class Queue
|
||||||
|
*/
|
||||||
|
class Queue
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* 当前实例
|
||||||
|
* @var Queue
|
||||||
|
*/
|
||||||
|
private static $instance;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Redis实例
|
||||||
|
* @var Redis
|
||||||
|
*/
|
||||||
|
private $redis;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 单例入口
|
||||||
|
* @param mixed ...$vars
|
||||||
|
* @return Queue
|
||||||
|
*/
|
||||||
|
public static function instance(...$vars): Queue
|
||||||
|
{
|
||||||
|
if (!self::$instance) {
|
||||||
|
self::$instance = new self(...$vars);
|
||||||
|
}
|
||||||
|
|
||||||
|
return self::$instance;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function __clone()
|
||||||
|
{
|
||||||
|
// 单例,私有__clone方法
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Queue constructor.
|
||||||
|
* @param $host
|
||||||
|
* @param int $port
|
||||||
|
* @param string $password
|
||||||
|
*/
|
||||||
|
private function __construct($host, $port = 6379, $password = '')
|
||||||
|
{
|
||||||
|
$redis = new Redis();
|
||||||
|
$redis->connect($host, $port);
|
||||||
|
$redis->auth($password);
|
||||||
|
$this->redis = $redis;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 添加任务到队列
|
||||||
|
* @param $key
|
||||||
|
* @param $value
|
||||||
|
* @param int $sec
|
||||||
|
* @return bool
|
||||||
|
*/
|
||||||
|
public function addDelayTask($key, $value, $sec = 10): bool
|
||||||
|
{
|
||||||
|
// 获取当前时间戳
|
||||||
|
$expireTime = $this->getTime($sec);
|
||||||
|
// 添加到有序集合
|
||||||
|
return $this->redis->zAdd($key, $expireTime, $value) > 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 删除延迟任务
|
||||||
|
* @param $key
|
||||||
|
* @param $value
|
||||||
|
* @return bool
|
||||||
|
*/
|
||||||
|
public function remDelayTask($key, $value): bool
|
||||||
|
{
|
||||||
|
// 从有序集合删除指定值
|
||||||
|
return $this->redis->zRem($key, $value) > 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 处理延迟任务
|
||||||
|
* @param $key
|
||||||
|
* @param $callback
|
||||||
|
* @param int $sleep
|
||||||
|
*/
|
||||||
|
public function handleDelayTask($key, $callback, $sleep = 500000)
|
||||||
|
{
|
||||||
|
// 死循环
|
||||||
|
while (true) {
|
||||||
|
// 尝试取出有序集合中第一个值
|
||||||
|
$task = $this->redis->zRange($key, 0, 0, true);
|
||||||
|
if (!empty($task)) {
|
||||||
|
$value = array_keys($task)[0];
|
||||||
|
$expireTime = $task[$value];
|
||||||
|
// 如果当前时间已经到达设定到执行时间
|
||||||
|
// 则开始执行逻辑,并删除当前任务
|
||||||
|
if ($this->getTime() >= $expireTime) {
|
||||||
|
call_user_func($callback, $value);
|
||||||
|
$this->remDelayTask($key, $value);
|
||||||
|
// 直接进入下一次循环,跳过usleep
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 减少读取次数
|
||||||
|
usleep($sleep);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 添加实时任务
|
||||||
|
* @param $key
|
||||||
|
* @param $value
|
||||||
|
* @return false|int
|
||||||
|
*/
|
||||||
|
public function addTask($key, $value)
|
||||||
|
{
|
||||||
|
// 添加到列表
|
||||||
|
return $this->redis->lPush($key, $value);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 删除实时任务
|
||||||
|
* @param $key
|
||||||
|
* @param $value
|
||||||
|
* @return bool|int
|
||||||
|
*/
|
||||||
|
public function remTask($key, $value)
|
||||||
|
{
|
||||||
|
// 从列表删除指定值
|
||||||
|
return $this->redis->lRem($key, $value, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 处理实时队列
|
||||||
|
* @param $key
|
||||||
|
* @param $callback
|
||||||
|
*/
|
||||||
|
public function handleTask($key, $callback)
|
||||||
|
{
|
||||||
|
// 死循环
|
||||||
|
while (true) {
|
||||||
|
// 阻塞读取列表中的值
|
||||||
|
$task = $this->redis->brPop($key, 0);
|
||||||
|
// 执行用户逻辑
|
||||||
|
call_user_func($callback, $task[1]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取现在时间戳(微秒)
|
||||||
|
* @param int $sec
|
||||||
|
* @return int
|
||||||
|
*/
|
||||||
|
private function getTime($sec = 0): int
|
||||||
|
{
|
||||||
|
return time() + $sec;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
36
README.en.md
36
README.en.md
|
@ -1,36 +0,0 @@
|
||||||
# Redis queue
|
|
||||||
|
|
||||||
#### Description
|
|
||||||
Redis实现消息队列
|
|
||||||
|
|
||||||
#### Software Architecture
|
|
||||||
Software architecture description
|
|
||||||
|
|
||||||
#### Installation
|
|
||||||
|
|
||||||
1. xxxx
|
|
||||||
2. xxxx
|
|
||||||
3. xxxx
|
|
||||||
|
|
||||||
#### Instructions
|
|
||||||
|
|
||||||
1. xxxx
|
|
||||||
2. xxxx
|
|
||||||
3. xxxx
|
|
||||||
|
|
||||||
#### Contribution
|
|
||||||
|
|
||||||
1. Fork the repository
|
|
||||||
2. Create Feat_xxx branch
|
|
||||||
3. Commit your code
|
|
||||||
4. Create Pull Request
|
|
||||||
|
|
||||||
|
|
||||||
#### Gitee Feature
|
|
||||||
|
|
||||||
1. You can use Readme\_XXX.md to support different languages, such as Readme\_en.md, Readme\_zh.md
|
|
||||||
2. Gitee blog [blog.gitee.com](https://blog.gitee.com)
|
|
||||||
3. Explore open source project [https://gitee.com/explore](https://gitee.com/explore)
|
|
||||||
4. The most valuable open source project [GVP](https://gitee.com/gvp)
|
|
||||||
5. The manual of Gitee [https://gitee.com/help](https://gitee.com/help)
|
|
||||||
6. The most popular members [https://gitee.com/gitee-stars/](https://gitee.com/gitee-stars/)
|
|
37
README.md
37
README.md
|
@ -1,37 +0,0 @@
|
||||||
# Redis queue
|
|
||||||
|
|
||||||
#### 介绍
|
|
||||||
Redis实现消息队列
|
|
||||||
|
|
||||||
#### 软件架构
|
|
||||||
软件架构说明
|
|
||||||
|
|
||||||
|
|
||||||
#### 安装教程
|
|
||||||
|
|
||||||
1. xxxx
|
|
||||||
2. xxxx
|
|
||||||
3. xxxx
|
|
||||||
|
|
||||||
#### 使用说明
|
|
||||||
|
|
||||||
1. xxxx
|
|
||||||
2. xxxx
|
|
||||||
3. xxxx
|
|
||||||
|
|
||||||
#### 参与贡献
|
|
||||||
|
|
||||||
1. Fork 本仓库
|
|
||||||
2. 新建 Feat_xxx 分支
|
|
||||||
3. 提交代码
|
|
||||||
4. 新建 Pull Request
|
|
||||||
|
|
||||||
|
|
||||||
#### 特技
|
|
||||||
|
|
||||||
1. 使用 Readme\_XXX.md 来支持不同的语言,例如 Readme\_en.md, Readme\_zh.md
|
|
||||||
2. Gitee 官方博客 [blog.gitee.com](https://blog.gitee.com)
|
|
||||||
3. 你可以 [https://gitee.com/explore](https://gitee.com/explore) 这个地址来了解 Gitee 上的优秀开源项目
|
|
||||||
4. [GVP](https://gitee.com/gvp) 全称是 Gitee 最有价值开源项目,是综合评定出的优秀开源项目
|
|
||||||
5. Gitee 官方提供的使用手册 [https://gitee.com/help](https://gitee.com/help)
|
|
||||||
6. Gitee 封面人物是一档用来展示 Gitee 会员风采的栏目 [https://gitee.com/gitee-stars/](https://gitee.com/gitee-stars/)
|
|
|
@ -0,0 +1,15 @@
|
||||||
|
<?php
|
||||||
|
date_default_timezone_set('PRC');
|
||||||
|
ini_set('default_socket_timeout', -1);
|
||||||
|
|
||||||
|
require './DelayTask.php';
|
||||||
|
|
||||||
|
$queue = Queue::instance('127.0.0.1', 6379, 'topnuomi');
|
||||||
|
$queue->handleDelayTask('order', function ($value) {
|
||||||
|
print '取出任务 ' . $value . ' [' . date('Y-m-d H:i:s') . ']' . PHP_EOL;
|
||||||
|
});
|
||||||
|
// $queue->handleTask('order', function ($value) {
|
||||||
|
// print '执行任务 ' . $value . ' [' . date('Y-m-d H:i:s') . ']' . PHP_EOL;
|
||||||
|
// // TODO 执行具体逻辑
|
||||||
|
// // sleep(2);
|
||||||
|
// });
|
|
@ -0,0 +1,20 @@
|
||||||
|
<?php
|
||||||
|
date_default_timezone_set('PRC');
|
||||||
|
|
||||||
|
require './DelayTask.php';
|
||||||
|
|
||||||
|
$queue = Queue::instance('127.0.0.1', 6379, 'topnuomi');
|
||||||
|
for ($i = 0; $i < 10; $i++) {
|
||||||
|
// 延迟任务
|
||||||
|
$taskName = uniqid();
|
||||||
|
$flag = $queue->addDelayTask('order', $taskName, 10);
|
||||||
|
|
||||||
|
// 实时任务
|
||||||
|
// $taskName = uniqid();
|
||||||
|
// $flag = $queue->addTask('order', json_encode([
|
||||||
|
// 'name' => $taskName,
|
||||||
|
// ], 256));
|
||||||
|
if ($flag) {
|
||||||
|
print '创建任务 ' . $i . ' [' . date('Y-m-d H:i:s') . ']' . PHP_EOL;
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue