记得在上一家公司时公司没有延迟队列,直接使用redis list进行使用,如果不到执行时间则继续丢回去,这样的方式太浪费IO,而且没办法保证执行顺序。本文没有使用有赞的redis延迟队列设计,使用的是redis有序集合的特性来完成。大致思路如下:
(1).下单成功通过zadd key score value命令把订单信息写入到集合中,例如
key:order score:指定要执行的时间戳(单位秒) value:订单id
集合的最终元素成员如下
score value 1603620459 202010250100 1603620460 202010250101
(2).通过zrangebyscore命令取需要执行的元素,例如
ZRANGEBYSCORE order (0 1603620500
例如查询score大于0,并且score小于当前时间戳的数据
(3).查询到数据我们应该从集合中删除此元素,使用zrem命令即可,如果删除的失败,说明已经被其他进程消费,可以丢弃。
我编写了一个PHP的实现。
<?php
class DelayQueue
{
/**
* name
* @var string|null
*/
public static $name = null;
/**
* Redis_Handler
* @var Redis|null
*/
public static $handler = null;
/**
* add
* @param int $score
* @param string $value
* @return int
*/
public static function add($score, $value)
{
return static::$handler->zAdd(static::$name, $score, $value);
}
/**
* get
* @param int $e_score
* @param int $s_score
* @param int $limit
* @param bool $remove
* @return array
*/
public static function get($e_score, $s_score = 0, $limit = 10, $remove = true)
{
$list = static::$handler->zRangeByScore(static::$name, $s_score, $e_score, ['limit' => [0, $limit]]);
if ($remove)
{
foreach ($list as $key => $value)
{
if (!static::del($value)) unset($list[$key]);
}
}
return $list;
}
/**
* del
* @param $value
* @return int
*/
public static function del($value)
{
return static::$handler->zRem(static::$name, $value);
}
}(4).生成者的代码:
//1.加载Redis
$redis = new Redis();
$redis->connect('127.0.0.1', 6379);
//2.设置延迟队列
DelayQueue::$name = 'order';
DelayQueue::$handler = $redis;
//3.投递队列
$time = time() + 30; //下单成功30秒需要处理
$orderId = uniqid();
DelayQueue::add($time, $orderId);(5).消费者代码:
//1.加载Redis
$redis = new Redis();
$redis->connect('127.0.0.1', 6379);
//2.设置延迟队列
DelayQueue::$name = 'order';
DelayQueue::$handler = $redis;
//3.查询时间小于当前时间的队列数据,默认输出10条
$time = time();
$list = DelayQueue::get($time);
foreach ($list as $value)
{
//输出信息
echo "订单Id:" . $value . PHP_EOL;
//发送消息给订单关联的用户(伪代码)
//sendMsgToUserByOrderId($value);
}(6).其他从队列取值的写法说明:
(1).在队列中查询time<=1603597141 and time>=0 的元素(同时队列中会自动删除符合条件的元素) $time = 1603597141; $list = DelayQueue::get($time); (2).在队列中查询time<=1603597141 and time>=1603597132 的元素(同时队列中会自动删除符合条件的元素) $time1 = 1603597413; $time2 = 1603597132; $list = DelayQueue::get($time1, $time2); (3).在队列中查询time<=1603597141 and time>=1603597132 的元素,只返回1条(同时队列中会自动删除这条元素) $time1 = 1603597555; $time2 = 1603597132; $list = DelayQueue::get($time1, $time2,1); (4).在队列中查询time<=1603597141 and time>=1603597132 的元素返回10条即可,不删除队列数据,仅查看数据 $time1 = 1603597693; $time2 = 1603597132; $list = DelayQueue::get($time1, $time2, 10, false);
(7).解决有序集合元素值不得重复的问题:
Redis有序集合中元素内容不得重复,上面实例中都是传递的订单Id,如果我们想投递多次相同订单Id,何如? (1).Value中传递唯一Id,同订单Id组合的Json形式,例如 $data = [ 'order_id'=>1, 'order_uniqid'=>uniqid() ]; $res = DelayQueue::add(time() + 30, json_encode($data)); (2).Value中前X位存储订单Id,后X位存储唯一Id(推荐)
有序集合是通过哈希表实现的,所以添加,删除,查找的复杂度都是 O(1)。 集合中最大的成员数为 232 - 1 (4294967295, 每个集合可存储40多亿个成员)。
gitee代码仓库:https://gitee.com/392223903/easy-delay-queue
源码:特别适用于微信支付中通知微信支付网关function array2xml($arr, $level = 1) { $s = $level == 1 ? "<xml&g...
php官方的超全局变量$_SERVER['PHP_SELF']也能直接获取,只不过如果url参数太多也会获取。下面提供一个方法获取/* * 获取当前PHP文件名称 */ if (!function_exists('phpself...
(1).前端文件:<form action="upload.php" method="post" enctype="multipart/form-data"> &...
(1).在PHP中可以查看的环境变量包括: (1.1).电脑环境变量 (2.1).服务器环境变量(2).getenv()函数获取一个环境变量的值.参数1是环境变量的key,参数2值为true的时候仅从你的电脑环境变量中查找,参数2值为false会从两种变量中全部查询//获取我电脑登录的用户名,输出A...
概念请参考w3school文章: redis watch ,redis exec (看完基本秒懂)(1)基本事务://连接本地的 Redis 服务 $redis = new Redis(); $redis->con...
(1).管道是干嘛的?管道是用于进程之间通信的,传播或交换信息(2).管道有几种?(2.1).匿名管道(pipe):管道是一种半双工的通信方式,数据只能单向流动,而且只能在具有亲缘关系的进程间使用。通常是指父子进程关系。(2.2).高级管道(popen):将另一个程序当做一个新的进程在当前程序进程中...