在秒杀环节中超卖一直是个大忌,我们允许一定程度上的少卖但是我们绝不允许出现超卖的现象.设想一下,那种一元抢购千元手机的秒杀活动,如果超卖了,损失肯定是比较大的.前段时间楼主做过一个类似于秒杀的项目,所以做了一下总结,当然实际项目要比我下面讲的要复杂.
1. 说明
Lua 是一种轻量小巧的脚本语言,采用C语言实现,一方面减小了访问redis的网络开销,另一方面天然的原子性有效的解决了事务的难题,但是redis集群对lua集群的支持有限的,集群有一个slot的概念,Redis要求单个Lua脚本操作的key必须在同一个节点上,但是Cluster会将数据自动分布到不同的节点(虚拟的16384个slot,具体看官方文档),但是集群模式下不同的key根据redis的hash算法会分布在不同的槽,集群是不支持多key查询的,解决办法是使用redis的hash tag,存入的时候把要多key查询的 key用{}括起来。keySlot算法中,如果key包含{},就会使用第一个{}内部的字符串作为hash key,这样就可以保证拥有同样{}内部字符串的key就会拥有相同slot。也可以使用代理来完成Lua在集群模式下的实现.本文redis是非集群模式.
2. 背景
现在我要实现这样一个功能,我们只有两件商品,商品hh初始化库存为7,商品newday初始化库存为6,现在用户每次都捆绑买这两件商品,只要有一个商品库存不足,本次活动结束.
3. 基于Redssion+Lua实现
redission实现了JDK中的Lock接口,所以使用方式上和我们用lock一样,只是Redssion的锁是分布式的,这也是redis官方现在比较力推的.
下面分享一下干货,毕竟talk is cheap.
逻辑控制层代码:
package com.example.demo.controller;
import org.redisson.api.RLock;
import org.redisson.api.RedissonClient;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.data.redis.core.script.DefaultRedisScript;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;
import javax.annotation.Resource;
import java.util.Arrays;
import java.util.List;
import java.util.concurrent.TimeUnit;
@RestController
public class RedissionTest {
@Resource
private RedissonClient redissonClient;
@Resource
private DefaultRedisScript<Boolean> redisScript;
@Resource
private StringRedisTemplate stringRedisTemplate;
@RequestMapping(value = "test", method = {RequestMethod.GET})
public void test() {
String ss="66666";
System.out.println(ss);
RLock lock = redissonClient.getLock(ss);
try {
// 1. 最常见的使用方法
//lock.lock();
// 2. 支持过期解锁功能,10秒钟以后自动解锁, 无需调用unlock方法手动解锁
//lock.lock(10, TimeUnit.SECONDS);
// 3. 尝试加锁,最多等待2秒,上锁以后8秒自动解锁
boolean res = lock.tryLock(2, 8, TimeUnit.SECONDS);
if (res) { //成功
//处理业务
System.out.println("33333333{} " + ss);
List<String> keys = Arrays.asList("hh","newday");
Boolean execute = stringRedisTemplate.execute(redisScript, keys);
System.out.println("999");
System.out.println("**** {}" + execute);
}
} catch (Exception e) {
e.printStackTrace();
} finally {
//释放锁
System.out.println("8888");
lock.unlock();
}
}
}
Redssion配置(本文以单机模式演示):
package com.example.demo.config;
import org.redisson.Redisson;
import org.redisson.api.RedissonClient;
import org.redisson.client.codec.StringCodec;
import org.redisson.config.Config;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
public class RedissonConfig {
@Bean
public RedissonClient getRedisson(){
Config config = new Config();
//单机模式 依次设置redis地址和密码
config.useSingleServer().
setAddress("redis://127.0.0.1:6379?auth=123456")
.setPassword("123456");
config.setCodec(new StringCodec());
return Redisson.create(config);
}
}
初始化Lua配置:
package com.example.demo.config;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.io.ClassPathResource;
import org.springframework.data.redis.core.script.DefaultRedisScript;
import org.springframework.scripting.support.ResourceScriptSource;
@Configuration
public class LuaConfiguration {
@Bean
public DefaultRedisScript<Boolean> redisScript() {
DefaultRedisScript<Boolean> redisScript = new DefaultRedisScript<>();
redisScript.setScriptSource(new ResourceScriptSource(new ClassPathResource("Test.lua")));
redisScript.setResultType(Boolean.class);
return redisScript;
}
}
3. 压力测试
这里我采用的压力测试工具为siege,在mac上面使用起来还是比较简单的.
为了有效的模拟实际的效果,并发50个线程循环5秒在线压力测试,5秒内的请求数远超过商品库存数.
3.1测试结果
从执行结果来看,我们有效的控制了超卖的现象.
Redis执行结果变化:
控制台执行过程:
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
33333333{} 66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
999
**** {}true
8888
66666
33333333{} 66666
66666
999
**** {}true
8888
33333333{} 66666
999
**** {}true
8888
66666
33333333{} 66666
66666
999
**** {}true
8888
66666
33333333{} 66666
999
**** {}true
8888
66666
33333333{} 66666
999
**** {}true
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
3333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
66666
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
66666
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{}
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
66666
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
66666
66666
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
66666
66666
66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false