Redis事务秒杀案例
- 本笔记转载与尚硅谷 (opens new window)笔记,可在Bilibili同步观看
- 尚硅谷-视频观看地址 (opens new window)
# Redis事务秒杀案例
# 解决计数器和人员记录的事务操作
# Redis 事务--秒杀并发模拟
使用工具 ab 模拟测试 CentOS6 默认安装 CentOS7 需要手动安装
# 联网:yum install httpd-tools
# 无网络
- 进入 cd /run/media/root/CentOS 7 x86_64/Packages(路径跟 centos6 不同)
- 顺序安装 apr-1.4.8-3.el7.x86_64.rpm apr-util-1.5.2-6.el7.x86_64.rpm httpd-tools-2.4.6-67.el7.centos.x86_64.rpm
# 测试及结果
# 通过 ab 测试
vim postfile 模拟表单提交参数,以&符号结尾;存放当前目录。 内容:prodid=0101& ab -n 2000 -c 200 -k -p ~/postfile -T application/x-www-form-urlencoded http://192.168.2.115:8081/Seckill/doseckill
# 超卖
# 超卖问题
# 利用乐观锁淘汰用户,解决超卖问题。
//增加乐观锁
jedis.watch(qtkey);
//3.判断库存
String qtkeystr = jedis.get(qtkey);
if(qtkeystr==null || "".equals(qtkeystr.trim())) {
System.out.println("未初始化库存");
jedis.close();
return false ;
}
int qt = Integer.parseInt(qtkeystr);
if(qt<=0) {
System.err.println("已经秒光");
jedis.close();
return false;
}
//增加事务
Transaction multi = jedis.multi();
//4.减少库存
//jedis.decr(qtkey);
multi.decr(qtkey);
//5.加人
//jedis.sadd(usrkey, uid);
multi.sadd(usrkey, uid);
//执行事务
List<Object> list = multi.exec();
//判断事务提交是否失败
if(list==null || list.size()==0) {
System.out.println("秒杀失败");
jedis.close();
return false;
}
System.err.println("秒杀成功");
jedis.close();
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
29
30
31
32
33
34
35
36
37
38
39
40
# 继续增加并发测试
11.5.1. 连接有限制 ab -n 2000 -c 200 -k -p postfile -T 'application/x-www-form-urlencoded' http://192.168.140.1:8080/seckill/doseckill
增加-r 参数,-r Don't exit on socket receive errors. ab -n 2000 -c 100 -r -p postfile -T 'application/x-www-form-urlencoded' http://192.168.140.1:8080/seckill/doseckill
# 已经秒光,可是还有库存
ab -n 2000 -c 100 -p postfile -T 'application/x-www-form-urlencoded' http://192.168.137.1:8080/seckill/doseckill 已经秒光,可是还有库存。原因,就是乐观锁导致很多请求都失败。先点的没秒到,后点的可能秒到了。
# 连接超时,通过连接池解决
# 连接池
节省每次连接 redis 服务带来的消耗,把连接好的实例反复利用。 通过参数管理连接的行为 代码见项目中
- 链接池参数
- MaxTotal:控制一个 pool 可分配多少个 jedis 实例,通过 pool.getResource()来获取;如果赋值为-1,则表示不限制;如果 pool 已经分配了 MaxTotal 个 jedis 实例,则此时 pool 的状态为 exhausted。
- maxIdle:控制一个 pool 最多有多少个状态为 idle(空闲)的 jedis 实例;
- MaxWaitMillis:表示当 borrow 一个 jedis 实例时,最大的等待毫秒数,如果超过等待时间,则直接抛 JedisConnectionException;
- testOnBorrow:获得一个 jedis 实例的时候是否检查连接可用性(ping());如果为 true,则得到的 jedis 实例均是可用的;
# 解决库存遗留问题
# LUA 脚本
Lua 是一个小巧的脚本语言,Lua 脚本可以很容易的被 C/C++ 代码调用,也可以反过来调用 C/C++的函数,Lua 并没有提供强大的库,一个完整的 Lua 解释器不过 200k,所以 Lua 不适合作为开发独立应用程序的语言,而是作为嵌入式脚本语言。 很多应用程序、游戏使用 LUA 作为自己的嵌入式脚本语言,以此来实现可配置性、可扩展性。 这其中包括魔兽争霸地图、魔兽世界、博德之门、愤怒的小鸟等众多游戏插件或外挂。 https://www.w3cschool.cn/lua/
# LUA 脚本在 Redis 中的优势
将复杂的或者多步的 redis 操作,写为一个脚本,一次提交给 redis 执行,减少反复连接 redis 的次数。提升性能。 LUA 脚本是类似 redis 事务,有一定的原子性,不会被其他命令插队,可以完成一些 redis 事务性的操作。 但是注意 redis 的 lua 脚本功能,只有在 Redis 2.6 以上的版本才可以使用。 利用 lua 脚本淘汰用户,解决超卖问题。 redis 2.6 版本以后,通过 lua 脚本解决争抢问题,实际上是 redis 利用其单线程的特性,用任务队列的方式解决多任务并发问题。
# Redis事务秒杀案例_代码
# 项目结构
# 第一版:简单版
老师点 10 次,正常秒杀 同学一起点试一试,秒杀也是正常的。这是因为还达不到并发的效果。 使用工具 ab 模拟并发测试,会出现超卖情况。查看库存会出现负数。
# 第二版:加事务-乐观锁(解决超卖),但出现遗留库存和连接超时
# 第三版:连接池解决超时问题
# 第四版:解决库存依赖问题,LUA 脚本
local userid=KEYS[1];
local prodid=KEYS[2];
local qtkey="sk:"..prodid..":qt";
local usersKey="sk:"..prodid.":usr';
local userExists=redis.call("sismember",usersKey,userid);
if tonumber(userExists)==1 then
return 2;
end
local num= redis.call("get" ,qtkey);
if tonumber(num)<=0 then
return 0;
else
redis.call("decr",qtkey);
redis.call("sadd",usersKey,userid);
end
return 1;
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16