使用PHP+Redis实现延迟任务,实现自动取消订单功能


Posted in PHP onNovember 21, 2019

简单定时任务解决方案:使用redis的keyspace notifications(键失效后通知事件) 需要注意此功能是在redis 2.8版本以后推出的,因此你服务器上的reids最少要是2.8版本以上;

(A)业务场景:

1、当一个业务触发以后需要启动一个定时任务,在指定时间内再去执行一个任务(如自动取消订单,自动完成订单等功能)

2、redis的keyspace notifications 会在key失效后发送一个事件,监听此事件的的客户端就可以收到通知

(B)服务准备:

1、修改reids配置文件(redis.conf)【window系统配置文件为:redis.windows.conf 】

redis默认不会开启keyspace notifications,因为开启后会对cpu有消耗

备注:E:keyevent事件,事件以__keyevent@<db>__为前缀进行发布;

x:过期事件,当某个键过期并删除时会产生该事件;

原配置为:

notify-keyspace-events ""

更改 配置如下:

notify-keyspace-events "Ex"

保存配置后,重启Redis服务,使配置生效

[root@chokingwin etc]#
service redis-server restart /usr/local/redis/etc/redis.conf 
Stopping redis-server: [ OK ] 
Starting redis-server: [ OK ]

window系统重启redis ,先切换到redis文件目录,然后关闭redis服务(redis-server --service-stop),再开启(redis-server --service-start)

使用PHP+Redis实现延迟任务,实现自动取消订单功能

C)文件代码:

phpredis实现订阅Keyspace notification,可实现自动取消订单,自动完成订单。以下为测试例子

创建4个文件,然后自行修改数据库和redis配置参数

db.class.php

<?php
class mysql
{
 private $mysqli;
 private $result;
 /**
 * 数据库连接
 * @param $config 配置数组
 */

 public function connect()
 {
 $config=array(
 'host'=>'127.0.0.1',
 'username'=>'root',
 'password'=>'168168',
 'database'=>'test',
 'port'=>3306,
 );

 $host = $config['host']; //主机地址
 $username = $config['username'];//用户名
 $password = $config['password'];//密码
 $database = $config['database'];//数据库
 $port = $config['port']; //端口号
 $this->mysqli = new mysqli($host, $username, $password, $database, $port);

 }
 /**
 * 数据查询
 * @param $table 数据表
 * @param null $field 字段
 * @param null $where 条件
 * @return mixed 查询结果数目
 */
 public function select($table, $field = null, $where = null)
 {
 $sql = "SELECT * FROM `{$table}`";
 //echo $sql;exit;
 if (!empty($field)) {
 $field = '`' . implode('`,`', $field) . '`';
 $sql = str_replace('*', $field, $sql);
 }
 if (!empty($where)) {
 $sql = $sql . ' WHERE ' . $where;
 }


 $this->result = $this->mysqli->query($sql);

 return $this->result;
 }
 /**
 * @return mixed 获取全部结果
 */
 public function fetchAll()
 {
 return $this->result->fetch_all(MYSQLI_ASSOC);
 }
 /**
 * 插入数据
 * @param $table 数据表
 * @param $data 数据数组
 * @return mixed 插入ID
 */
 public function insert($table, $data)
 {
 foreach ($data as $key => $value) {
 $data[$key] = $this->mysqli->real_escape_string($value);
 }
 $keys = '`' . implode('`,`', array_keys($data)) . '`';
 $values = '\'' . implode("','", array_values($data)) . '\'';
 $sql = "INSERT INTO `{$table}`( {$keys} )VALUES( {$values} )";
 $this->mysqli->query($sql);
 return $this->mysqli->insert_id;
 }
 /**
 * 更新数据
 * @param $table 数据表
 * @param $data 数据数组
 * @param $where 过滤条件
 * @return mixed 受影响记录
 */
 public function update($table, $data, $where)
 {
 foreach ($data as $key => $value) {
 $data[$key] = $this->mysqli->real_escape_string($value);
 }
 $sets = array();
 foreach ($data as $key => $value) {
 $kstr = '`' . $key . '`';
 $vstr = '\'' . $value . '\'';
 array_push($sets, $kstr . '=' . $vstr);
 }
 $kav = implode(',', $sets);
 $sql = "UPDATE `{$table}` SET {$kav} WHERE {$where}";

 $this->mysqli->query($sql);
 return $this->mysqli->affected_rows;
 }
 /**
 * 删除数据
 * @param $table 数据表
 * @param $where 过滤条件
 * @return mixed 受影响记录
 */
 public function delete($table, $where)
 {
 $sql = "DELETE FROM `{$table}` WHERE {$where}";
 $this->mysqli->query($sql);
 return $this->mysqli->affected_rows;
 }
}

index.php

<?php
require_once 'Redis2.class.php';
$redis = new \Redis2('127.0.0.1','6379','','15');
$order_sn = 'SN'.time().'T'.rand(10000000,99999999);
$use_mysql = 1; //是否使用数据库,1使用,2不使用
if($use_mysql == 1){
 /*
 * //数据表
 * CREATE TABLE `order` (
 * `ordersn` varchar(255) NOT NULL DEFAULT '',
 * `status` varchar(255) NOT NULL DEFAULT '',
 * `createtime` varchar(255) NOT NULL DEFAULT '',
 * `id` int(11) unsigned NOT NULL AUTO_INCREMENT,
 * PRIMARY KEY (`id`)
 * ) ENGINE=InnoDB AUTO_INCREMENT=27 DEFAULT CHARSET=utf8mb4;
 */
 require_once 'db.class.php';
 $mysql = new \mysql();
 $mysql->connect();
 $data = ['ordersn'=>$order_sn,'status'=>0,'createtime'=>date('Y-m-d H:i:s',time())];
 $mysql->insert('order',$data);
}
$list = [$order_sn,$use_mysql];
$key = implode(':',$list);
$redis->setex($key,3,'redis延迟任务'); //3秒后回调
$test_del = false; //测试删除缓存后是否会有过期回调。结果:没有回调
if($test_del == true){
 //sleep(1);
 $redis->delete($order_sn);
}
echo $order_sn;
/*
 * 测试其他key会不会有回调,结果:有回调
 * $k = 'test';
 * $redis2->set($k,'100');
 * $redis2->expire($k,10);
 *
*/

psubscribe.php

<?php
ini_set('default_socket_timeout', -1); //不超时
require_once 'Redis2.class.php';
$redis_db = '15';
$redis = new \Redis2('127.0.0.1','6379','',$redis_db);
// 解决Redis客户端订阅时候超时情况
$redis->setOption();
//当key过期的时候就看到通知,订阅的key __keyevent@<db>__:expired 这个格式是固定的,db代表的是数据库的编号,由于订阅开启之后这个库的所有key过期时间都会被推送过来,所以最好单独使用一个数据库来进行隔离
$redis->psubscribe(array('__keyevent@'.$redis_db.'__:expired'), 'keyCallback');
// 回调函数,这里写处理逻辑
function keyCallback($redis, $pattern, $channel, $msg)
{
 echo PHP_EOL;
 echo "Pattern: $pattern\n";
 echo "Channel: $channel\n";
 echo "Payload: $msg\n\n";
 $list = explode(':',$msg);
 $order_sn = isset($list[0])?$list[0]:'0';
 $use_mysql = isset($list[1])?$list[1]:'0';
 if($use_mysql == 1){
 require_once 'db.class.php';
 $mysql = new \mysql();
 $mysql->connect();
 $where = "ordersn = '".$order_sn."'";
 $mysql->select('order','',$where);
 $finds=$mysql->fetchAll();
 print_r($finds);
 if(isset($finds[0]['status']) && $finds[0]['status']==0){
 $data = array('status' => 3);
 $where = " id = ".$finds[0]['id'];
 $mysql->update('order',$data,$where);
 }
 }
}
//或者
/*$redis->psubscribe(array('__keyevent@'.$redis_db.'__:expired'), function ($redis, $pattern, $channel, $msg){
 echo PHP_EOL;
 echo "Pattern: $pattern\n";
 echo "Channel: $channel\n";
 echo "Payload: $msg\n\n";
 //................
});*/

Redis2.class.php

<?php
class Redis2
{
 private $redis;
 public function __construct($host = '127.0.0.1', $port = '6379',$password = '',$db = '15')
 {
 $this->redis = new Redis();
 $this->redis->connect($host, $port); //连接Redis
 $this->redis->auth($password); //密码验证
 $this->redis->select($db); //选择数据库
 }
 public function setex($key, $time, $val)
 {
 return $this->redis->setex($key, $time, $val);
 }
 public function set($key, $val)
 {
 return $this->redis->set($key, $val);
 }
 public function get($key)
 {
 return $this->redis->get($key);
 }
 public function expire($key = null, $time = 0)
 {
 return $this->redis->expire($key, $time);
 }
 public function psubscribe($patterns = array(), $callback)
 {
 $this->redis->psubscribe($patterns, $callback);
 }
 public function setOption()
 {
 $this->redis->setOption(\Redis::OPT_READ_TIMEOUT, -1);
 }
 public function lRange($key,$start,$end)
 {
 return $this->redis->lRange($key,$start,$end);
 }
 public function lPush($key, $value1, $value2 = null, $valueN = null ){
 return $this->redis->lPush($key, $value1, $value2 = null, $valueN = null );
 }
 public function delete($key1, $key2 = null, $key3 = null)
 {
 return $this->redis->delete($key1, $key2 = null, $key3 = null);
 }
}

window系统测试方法:先在cmd命令界面运行psubscribe.php,然后网页打开index.php。

使监听后台始终运行(订阅)

有个问题 做到这一步,利用 phpredis 扩展,成功在代码里实现对过期 Key 的监听,并在 psCallback()里进行回调处理。开头提出的两个需求已经实现。可是这里有个问题:redis 在执行完订阅操作后,终端进入阻塞状态,需要一直挂在那。且此订阅脚本需要人为在命令行执行,不符合实际需求。

实际上,我们对过期监听回调的需求,是希望它像守护进程一样,在后台运行,当有过期事件的消息时,触发回调函数。使监听后台始终运行 希望像守护进程一样在后台一样,

我是这样实现的。

Linux中有一个nohup命令。功能就是不挂断地运行命令。同时nohup把脚本程序的所有输出,都放到当前目录的nohup.out文件中,如果文件不可写,则放到<用户主目录>/nohup.out 文件中。那么有了这个命令以后,不管我们终端窗口是否关闭,都能够让我们的php脚本一直运行。

编写psubscribe.php文件:

<?php
#! /usr/bin/env php
ini_set('default_socket_timeout', -1); //不超时
require_once 'Redis2.class.php';
$redis_db = '15';
$redis = new \Redis2('127.0.0.1','6379','',$redis_db);
// 解决Redis客户端订阅时候超时情况
$redis->setOption();
//当key过期的时候就看到通知,订阅的key __keyevent@<db>__:expired 这个格式是固定的,db代表的是数据库的编号,由于订阅开启之后这个库的所有key过期时间都会被推送过来,所以最好单独使用一个数据库来进行隔离
$redis->psubscribe(array('__keyevent@'.$redis_db.'__:expired'), 'keyCallback');
// 回调函数,这里写处理逻辑
function keyCallback($redis, $pattern, $channel, $msg)
{
 echo PHP_EOL;
 echo "Pattern: $pattern\n";
 echo "Channel: $channel\n";
 echo "Payload: $msg\n\n";
 $list = explode(':',$msg);
 $order_sn = isset($list[0])?$list[0]:'0';
 $use_mysql = isset($list[1])?$list[1]:'0';
 if($use_mysql == 1){
 require_once 'db.class.php';
 $mysql = new \mysql();
 $mysql->connect();
 $where = "ordersn = '".$order_sn."'";
 $mysql->select('order','',$where);
 $finds=$mysql->fetchAll();
 print_r($finds);
 if(isset($finds[0]['status']) && $finds[0]['status']==0){
 $data = array('status' => 3);
 $where = " id = ".$finds[0]['id'];
 $mysql->update('order',$data,$where);
 }
 }
}
//或者
/*$redis->psubscribe(array('__keyevent@'.$redis_db.'__:expired'), function ($redis, $pattern, $channel, $msg){
 echo PHP_EOL;
 echo "Pattern: $pattern\n";
 echo "Channel: $channel\n";
 echo "Payload: $msg\n\n";
 //................
});*/

注意:我们在开头,申明 php 编译器的路径:

#! /usr/bin/env php

这是执行 php 脚本所必须的。

然后,nohup 不挂起执行 psubscribe.php,注意 末尾的 &

[root@chokingwin HiGirl]# nohup ./psubscribe.php & 
[1] 4456 nohup: ignoring input and appending output to `nohup.out'

说明:脚本确实已经在 4456 号进程上跑起来。

查看下nohup.out cat 一下 nohuo.out,看下是否有过期输出:

[root@chokingwin HiGirl]# cat nohup.out 
Pattern:__keyevent@0__:expired 
Channel: __keyevent@0__:expired 
Payload: name

运行index.php ,3秒后效果如上即成功

遇到问题:使用命令行模式开启监控脚本 ,一段时间后报错 :Error while sending QUERY packet. PID=xxx

解决方法:由于等待消息队列是一个长连接,而等待回调前有个数据库连接,数据库的wait_timeout=28800,所以只要下一条消息离上一条消息超过8小时,就会出现这个错误,把wait_timeout设置成10,并且捕获异常,发现真实的报错是 MySQL server has gone away ,
所以只要处理完所有业务逻辑后主动关闭数据库连接,即数据库连接主动close掉就可以解决问题

yii解决方法如下:

Yii::$app->db->close();

查看进程方法:

ps -aux|grep psubscribe.php

a:显示所有程序
u:以用户为主的格式来显示
x:显示所有程序,不以终端机来区分

查看jobs进程ID:[ jobs -l ]命令

www@iZ232eoxo41Z:~/tinywan $ jobs -l
[1]- 1365 Stopped (tty output) sudo nohup psubscribe.php > /dev/null 2>&1 
[2]+ 1370 Stopped (tty output) sudo nohup psubscribe.php > /dev/null 2>&1

终止后台运行的进程方法:

kill -9 进程号

清空 nohup.out文件方法:

cat /dev/null > nohup.out

我们在使用nohup的时候,一般都和&配合使用,但是在实际使用过程中,很多人后台挂上程序就这样不管了,其实这样有可能在当前账户非正常退出或者结束的时候,命令还是自己结束了。

所以在使用nohup命令后台运行命令之后,我们需要做以下操作:

1.先回车,退出nohup的提示。

2.然后执行exit正常退出当前账户。
3.然后再去链接终端。使得程序后台正常运行。

我们应该每次都使用exit退出,而不应该每次在nohup执行成功后直接关闭终端。这样才能保证命令一直在后台运行。

总结

以上所述是小编给大家介绍的使用PHP+Redis实现延迟任务,实现自动取消订单功能,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!
如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!

PHP 相关文章推荐
PHP调用三种数据库的方法(2)
Oct 09 PHP
一个php作的文本留言本的例子(一)
Oct 09 PHP
模拟SQLSERVER的两个函数:dateadd(),datediff()
Oct 09 PHP
php strtotime 函数UNIX时间戳
Jan 14 PHP
Php 构造函数construct的前下划线是双的_
Dec 08 PHP
php 数组动态添加实现代码(最土团购系统的价格排序)
Dec 30 PHP
PHP网页游戏学习之Xnova(ogame)源码解读(十四)
Jun 26 PHP
服务器上配置PHP运行环境教程
Feb 12 PHP
Windows7下的php环境配置教程
Feb 28 PHP
PHP实现的线索二叉树及二叉树遍历方法详解
Apr 25 PHP
php实现等比例压缩图片
Jul 26 PHP
Laravel中validation验证 返回中文提示 全局设置的方法
Sep 29 PHP
PHP框架实现WebSocket在线聊天通讯系统
Nov 21 #PHP
PHP读取Excel内的图片(phpspreadsheet和PHPExcel扩展库)
Nov 19 #PHP
使用PHP开发留言板功能
Nov 19 #PHP
关于Laravel参数验证的一些疑与惑
Nov 19 #PHP
php传值和传引用的区别点总结
Nov 19 #PHP
php 使用 __call实现重载功能示例
Nov 18 #PHP
PHP中通过getopt解析GNU C风格命令行选项
Nov 18 #PHP
You might like
用PHP开发GUI
2006/10/09 PHP
PHP的面试题集,附我的答案和分析(一)
2006/11/19 PHP
PHP 应用程序的安全 -- 不能违反的四条安全规则
2006/11/26 PHP
解析mysql left( right ) join使用on与where筛选的差异
2013/06/18 PHP
PHP使用gmdate实现将一个UNIX 时间格式化成GMT文本的方法
2015/03/19 PHP
php array_merge函数使用需要注意的一个问题
2015/03/30 PHP
php curl 模拟登录并获取数据实例详解
2016/12/22 PHP
详解PHP5.6.30与Apache2.4.x配置
2017/06/02 PHP
如何解决PHP获取不到SESSION信息之一般情况
2019/10/10 PHP
jquery 学习之二 属性 文本与值(text,val)
2010/11/25 Javascript
在IE和VB中支持png图片透明效果的实现方法(vb源码打包)
2011/04/01 Javascript
基于jQuery实现图片的前进与后退功能
2013/04/24 Javascript
JavaScript前端图片加载管理器imagepool使用详解
2014/12/29 Javascript
JavaScript中对象介绍
2014/12/31 Javascript
jquery实现横向图片轮播特效代码分享
2015/11/19 Javascript
基于javascript代码实现通过点击图片显示原图片
2015/11/29 Javascript
Javascript基于jQuery UI实现选中区域拖拽效果
2016/11/25 Javascript
JS 中可以提升幸福度的小技巧(可以识别更多另类写法)
2018/07/28 Javascript
详解在React中跨组件分发状态的三种方法
2018/08/09 Javascript
Vue2.0 v-for filter列表过滤功能的实现
2018/09/07 Javascript
Javascript读写cookie的实例源码
2019/03/16 Javascript
Vue Cli 3项目使用融云IM实现聊天功能的方法
2019/04/19 Javascript
[43:24]VG vs Serenity 2018国际邀请赛小组赛BO2 第二场 8.17
2018/08/20 DOTA
python使用正则表达式检测密码强度源码分享
2014/06/11 Python
Python3.6简单反射操作示例
2018/06/14 Python
CSS3教程(6):创建网站多列
2009/04/02 HTML / CSS
工程造价与管理专业应届生求职信
2013/11/23 职场文书
2014年巴西世界杯口号
2014/06/05 职场文书
检查机关党的群众路线个人整改措施
2014/10/04 职场文书
地震捐款简报
2015/07/21 职场文书
《角的初步认识》教学反思
2016/02/17 职场文书
《猴王出世》教学反思
2016/02/23 职场文书
字典算法实现及操作 --python(实用)
2021/03/31 Python
Python 高级库15 个让新手爱不释手(推荐)
2021/05/15 Python
Python利用zhdate模块实现农历日期处理
2022/03/31 Python
浅析Python OpenCV三种滤镜效果
2022/04/11 Python