消息队列可以使用MySQL来实现,可以参考博客PHP使用MySQL实现消息队列,虽然用MySQL可以实现,但是一般不这么用,因为MySQL的数据都存在硬盘中,而从硬盘中对MySQL的操作,I/O花费的代价很大,所以一般使用缓存来实现,因为缓存的数据是在内存中,访问内存的速度远快于访问硬盘的速度。另一方面,Redis有list类型的数据结构,非常适合做消息队列。

这里举一个很简单的秒杀例子:秒杀的名额只有5个,即消息队列的长度为5,名额已经满了之后,通知后来的人已经秒杀结束。然后后台会从消息队列中读取数据,然后将数据存到数据库中。因为消息队列长度只有5个,而且秒杀的那短短1,2秒并没有直接操作数据库,所以对于数据库来说,并没有什么压力。

先看一下数据库表(seckill)的结构:

mysql> desc seckill;
+----------+---------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+----------+---------+------+-----+---------+----------------+
| id | int(11) | NO | PRI | NULL | auto_increment |
| order_id | int(11) | NO | | NULL | |
| mobile | int(8) | YES | | 8888888 | |
+----------+---------+------+-----+---------+----------------+
3 rows in set (0.11 sec)

  然后是进行秒杀的用户程序(user.php),为了模拟,这里使用for循环来实现在短时间内发起大量的请求,但是要知道这是不准确的。

<?php
$redis=new Redis();
$redis->connect("127.0.0.1",6379); $key="seckill";
for($i=0;$i<10;$i++){
$order_id=rand(100000,999999);
$mobile=rand(11111111,99999999);
$value=$order_id."#".$mobile;//连接之后作为值
if($redis->llen("seckill") <5 ){
echo "秒杀成功,订单号为$order_id, 手机号为$mobile\n";
$redis->lpush($key,$value);
} else {
echo "秒杀已经结束\n";
}
}
?>

  运行结果:

[root@localhost ~]# php user.php
秒杀成功,订单号为643275, 手机号为50104929
秒杀成功,订单号为393012, 手机号为31213041
秒杀成功,订单号为994790, 手机号为23107569
秒杀成功,订单号为186135, 手机号为36549273
秒杀成功,订单号为821972, 手机号为11217760
秒杀已经结束
秒杀已经结束
秒杀已经结束
秒杀已经结束
秒杀已经结束

  

然后是后台程序将redis中订单读出,处理后存进数据库。

<?php
$redis=new Redis();
$redis->connect("127.0.0.1",6379);
$pdo=new PDO("mysql:host=localhost;dbname=test","root","root");
$pdo->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION); $stmt=$pdo->prepare("insert into seckill(id,order_id,mobile) values(?,?,?)");
$key="seckill"; while($redis->llen($key)){
//因为消息队列中添加消息是使用lpush,所以这里使用rpop
$order=$redis->rpop($key);
list($order_id,$mobile)=explode("#",$order);
echo "正在处理订单$order_id\t";
try{
$res=$stmt->execute(array(null,$order_id,$mobile));
if(!$res){
throw new PDOException("wrong");
}
} catch (PDOException $e){
echo $e->getMessage();
echo "订单处理失败\n";
$redis->rpush($key,$order_id."#".$mobile);//将数据恢复达到队列中
continue;
}
echo "订单处理完成\n";
}
?>

  运行:

[root@localhost ~]# php consumer.php
正在处理订单643275 订单处理完成
正在处理订单393012 订单处理完成
正在处理订单994790 订单处理完成
正在处理订单186135 订单处理完成
正在处理订单821972 订单处理完成

  查看数据库:

mysql> select * from seckill;
+----+----------+----------+
| id | order_id | mobile |
+----+----------+----------+
| 1 | 643275 | 50104929 |
| 2 | 393012 | 31213041 |
| 3 | 994790 | 23107569 |
| 4 | 186135 | 36549273 |
| 5 | 821972 | 11217760 |
+----+----------+----------+
5 rows in set (0.00 sec)

  

PHP使用Redis实现消息队列的更多相关文章

  1. Redis 做消息队列

    一般来说,消息队列有两种场景,一种是发布者订阅者模式,一种是生产者消费者模式.利用redis这两种场景的消息队列都能够实现.定义: 生产者消费者模式:生产者生产消息放到队列里,多个消费者同时监听队列, ...

  2. Redis作为消息队列服务场景应用案例

    NoSQL初探之人人都爱Redis:(3)使用Redis作为消息队列服务场景应用案例   一.消息队列场景简介 “消息”是在两台计算机间传送的数据单位.消息可以非常简单,例如只包含文本字符串:也可以更 ...

  3. redis resque消息队列

    Resque 目前正在学习使用resque .resque-scheduler来发布异步任务和定时任务,为了方便以后查阅,所以记录一下. resque和resque-scheduler其优点在于功能比 ...

  4. 【springboot】【redis】springboot+redis实现发布订阅功能,实现redis的消息队列的功能

    springboot+redis实现发布订阅功能,实现redis的消息队列的功能 参考:https://www.cnblogs.com/cx987514451/p/9529611.html 思考一个问 ...

  5. 【Redis】php+redis实现消息队列

    在项目中使用消息队列一般是有如下几个原因: 把瞬间服务器的请求处理换成异步处理,缓解服务器的压力 实现数据顺序排列获取 redis实现消息队列步骤如下: 1).redis函数rpush,lpop 2) ...

  6. Lumen开发:结合Redis实现消息队列(1)

    1.简介 Lumen队列服务为各种不同的后台队列提供了统一的API.队列允许你推迟耗时任务(例如发送邮件)的执行,从而大幅提高web请求速度. 1.1 配置 .env文件的QUEUE_DRIVER选项 ...

  7. Redis除了做缓存--Redis做消息队列/Redis做分布式锁/Redis做接口限流

    1.用Redis实现消息队列 用命令lpush入队,rpop出队 Long size = jedis.lpush("QueueName", message);//返回存放的数据条数 ...

  8. sping+redis实现消息队列的乱码问题

    使用spring支持redis实现消息队列,参考官方样例:https://spring.io/guides/gs/messaging-redis/ 实现后在运行过程中发现消费者在接收消息时会出现乱码的 ...

  9. 程序员过关斩将--redis做消息队列,香吗?

    Redis消息队列 在程序员这个圈子打拼了太多年,见过太多的程序员使用redis,其中一部分喜欢把redis做缓存(cache)使用,其中最典型的当属存储用户session,除此之外,把redis作为 ...

  10. NoSQL初探之人人都爱Redis:(3)使用Redis作为消息队列服务场景应用案例

    一.消息队列场景简介 “消息”是在两台计算机间传送的数据单位.消息可以非常简单,例如只包含文本字符串:也可以更复杂,可能包含嵌入对象.消息被发送到队列中,“消息队列”是在消息的传输过程中保存消息的容器 ...

随机推荐

  1. 为什么MySQL数据库索引选择使用B+树?

    在进一步分析为什么MySQL数据库索引选择使用B+树之前,我相信很多小伙伴对数据结构中的树还是有些许模糊的,因此我们由浅入深一步步探讨树的演进过程,在一步步引出B树以及为什么MySQL数据库索引选择使 ...

  2. forever start app.js 启动node时,服务访问一次后第二次就不能访问了

    开始总是找不到原因,是因为在启动服务时,没有设置日志文件.突然想到了是不是forever安装的有问题,就重新安装forever , 这时候提示系统 no space left on device  , ...

  3. Problem UVA1572-Self-Assembly(拓扑排序)

    Problem UVA1572-Self-Assembly Accept: 196  Submit: 1152 Time Limit: 3000 mSec Problem Description Au ...

  4. day16 Python map函数

    num_l=[1,2,10,5,3,7] #lambda x:x+1 # def add_one(x): # return x+1 #lambda x:x+1 # def reduce_one(x): ...

  5. python3 day02 大纲

    一. 格式化输出 %s 字符串的占位 %d 数字的占位 digit %f 浮点数 字符串 % (数据) 模板字符串(3.5) # username = "admin"# passw ...

  6. php使用MPDF导出PDF文件自定义字体

    最近公司要生成一个PDF文件,网上找了很多类库,使用mpdf最方便,通过HTML输入调整样式,官方地址:http://www.mpdf1.com/,更改字体这个问题困扰了我很久,网上找的方法都不适用, ...

  7. @synchronized深入理解

    @synchronized是线程同步锁,易用.可读性高. @synchronized(self) { 临界区 } 利用如下命令将其重写 clang -rewrite-objc file  得到C++实 ...

  8. 【Codeforces Round 1129】Alex Lopashev Thanks-Round (Div. 1)

    Codeforces Round 1129 这场模拟比赛做了\(A1\).\(A2\).\(B\).\(C\),\(Div.1\)排名40. \(A\)题是道贪心,可以考虑每一个站点是分开来的,把目的 ...

  9. Android学习之触点事件的处理

    知识点: 1. Android开发中的运动事件:触摸屏(TouchScreen)和滚动球(TrackBall) 2.对运动事件的处理:MotionEvent 3.触摸时必发的三个MotionEvent ...

  10. JAVA中map的分类和各自的特性

    java为数据结构中的映射定义了一个接口java.util.Map,他实现了四个类,分别是:HashMap,HashTable,LinkedHashMapTreeMap Map不允许键重复,但允许值重 ...