PHP【Laravel】delayer基于redis的实现订单超时改变状态
实现这个功能前你需要知道以下,不然可能会比较吃力:
1.服务器的计划任务,shell脚本,或者你有宝塔自带的计划任务会方便很多。
2.有所了解Redis。
3.会写PHP业务逻辑。
好了进入在正题,这里使用一个库delayer。
它是 基于 Redis 的延迟队列中间件,采用 Golang 开发,支持 PHP、Golang 等多种语言客户端。
因此在你的项目中,你需要为PHP装上redis扩展。注意不是Laravel的predis。就是你phpinfo()能看到有redis这个扩展。
之后为你的项目加上delayer
你的服务端需要:(就是在服务器中装上delayer,然后在服务器上启动)
https://github.com/mix-basic/delayer
你的客户端需要:(就是在你的项目中装上delayer类库,让你的代码能到delayer的四个基本操作)
https://github.com/mix-basic/delayer-client-php
然后总结一下client这边四个方法:
1.push(放入任务)
//配置好连接redis的信息
$client = new \Delayer\Client(config('database.redis.default'));
//data为你要存入的数据
$data = [
'orderID' => '20181017125789566488854744',
'action' => 'close',
];
//发送信息
$message = new Message([
// 任务ID,必须全局唯一
'id' => md5(uniqid(mt_rand(), true)),
// 主题,取出任务时需使用
'topic' => 'close_order',
// 必须转换为string类型
'body' => json_encode($data),
]);
/
* push($message, 10, 30)
* 第2个参数为延迟时间,第3个参数为延迟到期后如果任务没有被消费的最大生存时间(秒)
* 延迟时间:参数2设置时间到了才能pop出任务,否则pop返回false;
* 最大生存时间:参数2的时间到之后。倒计时参数3的时间。超时再pop返回false。
*
/
$ret = $client->push($message, 10, 30);
var_dump($ret);
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
2.pop(取到到期任务)
/
* 配置好连接redis的信息
* 要与Delayer服务器端配置的redis信息相同
/
$client = new \Delayer\Client(config('database.redis.default'));
//根据之前的topic值取出最先放入且没有消亡的任务
$message = $client->pop('close_order');
1
2
3
4
5
6
7
存在则返回如下:
任务被pop获取一次后就消亡了。再pop的话返回false或者是下一个到期任务。
3.bPop(阻塞取出)
$message = $client->bPop('close_order', 1);//这里相当于1秒后执行pop
1
4.remove(移除任务)
// push时定义的任务ID
$id = '***';
$ret = $client->remove($id);
var_dump($ret);
1
2
3
4
移出一个未到期任务。成功true,到期或消亡返回false。
----------------------------------------------以上是delayer的操作说明---------------------------------------------------
下面说明业务逻辑思路:
1.首先在服务器执行计划任务(定时任务)。这里我以宝塔作为例子。如下图:
shell脚本内容:
step=10
for (( i = 0; i < 60; i=(i+step) )); do
/www/server/php/73/bin/php /www/wwwroot/phal_admin/autoTask/deal_turn_format_list_result.php
sleep $step
done
exit 0
1
2
3
4
5
6
脚本说明
step =10 //代表10秒运行一次。
do 路径一(服务器如linux中php的可执行文件) 路径二(PHP的delayer取消订单的业务逻辑)
路径二的示例代码:
<?php
use App\Models\ShopOrder; //你的订单模型
use Delayer\Client; //delayer
require '/home/vagrant/code/public/init.php'; //必须项目入口文件,我这里是laravel
$client = new \Delayer\Client(config('database.redis.default')); //配置参数,不明白说明你没好好看
for ($x=0; $x<10; $x++) {
$result = $client->pop('close_order');
if(!$result ){ //服务器一直执行pop,否则break
break;
}
ShopOrder::cancelOrder($result->id); //如果pop到了超时订单就取消订单状态。这里自定义你的业务。
echo('订单超时取消!');
}
?>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
完成了以上恭喜你基本完成功能的大部分,此时你的服务器已经会自动对你的超时订单进行处理。
2.项目中的业务逻辑
现在服务器已经会对你的超时订单进行处理。
你只需要在未付款订单生成的时候push任务,用户付款的时候remove任务。
下面上伪代码:
未付款订单生成时(push入任务)
$data = [
//自定义内容
];
//发送信息
$message = new Message([
// 任务ID,必须全局唯一,这里可以放订单id(业务中remove需要)
'id' => 32,
// 主题,取出任务时需使用(服务器pop需要)
'topic' => 'close_order',
// 必须转换为string类型
'body' => json_encode($data),
]);
//放入任务,例如你的需求是30分钟自动取消订单。那么就是1800秒。后面30是消亡时间,自定义。
push($message, 1800, 30)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
用户付款之后(从任务中remove)
$client->remove($id); //message里你放入的id,如:上面的32
1
这样一个简单订单超时处理系统算是完成了。
如有不足或错误欢迎指出~
---------------------
PHP【Laravel】delayer基于redis的实现订单超时改变状态的更多相关文章
- Delayer 基于 Redis 的延迟消息队列中间件
Delayer 基于 Redis 的延迟消息队列中间件,采用 Golang 开发,支持 PHP.Golang 等多种语言客户端. 参考 有赞延迟队列设计 中的部分设计,优化后实现. 项目链接:http ...
- Matching Engine For Laravel(基于redis的撮合引擎),PHP高性能撮合引擎
Laravel Package for Matching Engine 快速开始 github地址 安装: composer require sting_bo/mengine 复制配置文件: php ...
- 基于 Redis 生成分布式订单号
环境依赖: //spingBoot <version>2.6.6</version> //jdk11 <dependency> <groupId>org ...
- 基于redis的订单号生成方案
目前,比较火的nosql数据库,如MongoDB,Redis,Riak都提供了类似incr原子行操作. 下面是PHP版的一种实现方式: <?php /** * 基于Redis的全局订单号id * ...
- [原创]Laravel 基于redis队列的解析
目录 参考链接 本文环境 为什么使用队列 Laravel 中的队列 分发任务 任务队列 Worker Last-Modified: 2019年5月10日11:44:18 参考链接 使用 Laravel ...
- [转载] 基于Redis实现分布式消息队列
转载自http://www.linuxidc.com/Linux/2015-05/117661.htm 1.为什么需要消息队列?当系统中出现“生产“和“消费“的速度或稳定性等因素不一致的时候,就需要消 ...
- Spring+Shiro搭建基于Redis的分布式权限系统(有实例)
摘要: 简单介绍使用Spring+Shiro搭建基于Redis的分布式权限系统. 这篇主要介绍Shiro如何与redis结合搭建分布式权限系统,至于如何使用和配置Shiro就不多说了.完整实例下载地址 ...
- 基于Redis实现延时队列服务
背景 在业务发展过程中,会出现一些需要延时处理的场景,比如: a.订单下单之后超过30分钟用户未支付,需要取消订单 b.订单一些评论,如果48h用户未对商家评论,系统会自动产生一条默认评论 c.点我达 ...
- 【转】基于Redis实现延时队列服务
背景 在业务发展过程中,会出现一些需要延时处理的场景,比如: a.订单下单之后超过30分钟用户未支付,需要取消订单b.订单一些评论,如果48h用户未对商家评论,系统会自动产生一条默认评论c.点我达订单 ...
随机推荐
- [CSP-S模拟测试]:阴阳(容斥+计数+递推)
题目传送门(内部题16) 输入格式 第一行两个整数$n$和$m$,代表网格的大小.接下来$n$行每行一个长度为$m$的字符串,每个字符若为$W$代表这个格子必须为阳,若为$B$代表必须为阴,若为$?$ ...
- ElasticSearch 简介概念及核心
1.ES是什么 ES是面向文档的Nosql,这意味着它可以存储整个对象或文档.然而它不仅仅是存储,还会索引(index)每个文档的内容使之可以被搜索.在es中,你可以对文档(而非成行成列的数据)进行索 ...
- (三)修改内核大小,适配目标板Nand flash分区配置
一. 修改内核大小 1. 在你的配置文件下uboot/include/config/xxx.h 里面有一个宏定义 #define MTDPARTS_DEFAULT "mtdparts=jz2 ...
- 用 Flask 来写个轻博客 (18) — 使用工厂模式来生成应用对象
Blog 项目源码:https://github.com/JmilkFan/JmilkFan-s-Blog 目录 目录 前文列表 工厂模式 使用工厂方法 Factory Method 创建 app 对 ...
- PHP-Redis扩展安装(四)
PHP-Redis扩展安装(四) 安装环境链接:http://pan.baidu.com/s/1i4IbJox Memecached 服务器安装(一) memcached php扩展(二) redis ...
- 【C#学习笔记】 List.AddRange 方法
[官方笔记] 将指定集合的元素添加到 List 的末尾 命名空间:System.Collections.Generic程序集:mscorlib(在 mscorlib.dll 中) public: vo ...
- electron测试TCP通信
这几天学习了一下Elctron,对于这个应用有了一点简单的认识,将这个过程记录一下. 首先,electron会加载main.js,在这里将整个程序启动,相当于其他程序的main函数了. 我是基于ele ...
- 50.Maximal Square(最大正方形)
Level Medium 题目描述: Given a 2D binary matrix filled with 0's and 1's, find the largest square conta ...
- JavaScript如何诞生
JavaScript之父谈语言诞生记 发表于2011-06-27 10:30| 9749次阅读| 来源ruanyifeng.com| 0 条评论| 作者阮一峰 prototypeprimitiveja ...
- 五、bootstrap-fileinput
一.bootstrap-fileinput <!DOCTYPE html> <html lang="en"> <head> <meta c ...