java redis 乐观锁_redis乐观锁实现高并发秒杀

在限量秒杀抢购的场景,一定会遇到抢购成功数超过限量的问题和高并发的情况影响系统性能

1、虽然能用数据库的锁避免,超过限量的问题。但是在大并发的情况下,大大影响数据库性能

2、为了避免并发操作数据库,我们可以使用队列来限制,但是并发量会让队列内存瞬间升高

3、我们又可以用悲观锁来实现,但是这样会造成用户等待,响应慢体验不好

因此我们可以利用redis来实现乐观锁

1、利用redis的watch功能,监控这个redisKey的状态值

2、获取redisKey的值

3、创建redis事务

4、给这个key的值+1

5、然后去执行这个事务,如果key的值被修改过则回滚,key不+1

这里我用java实现,我开20个线程模拟10000个人并发来抢购,其它语言也是一样的实现原理。

public static void main(String[] arg){

String redisKey = "redisTest";

ExecutorService executorService = Executors.newFixedThreadPool(20);

try {

Jedis jedis = new Jedis("47.107.221.219",6379);

jedis.set(redisKey,"0");

jedis.close();

}catch (Exception e){

e.printStackTrace();

}

for (int i=0;i<10000;i++){

executorService.execute(()->{

Jedis jedis1 = new Jedis("47.107.221.219",6379);

try {

jedis1.watch(redisKey);

String redisValue = jedis1.get(redisKey);

int valInteger = Integer.valueOf(redisValue);

String userInfo = UUID.randomUUID().toString();

if (valInteger<20){

Transaction transaction = jedis1.multi();

transaction.incr(redisKey);

List list = tx.exec();

if (list!=null){

System.out.println("用户:"+userInfo+",秒杀成功!当前成功人数:"+(valInteger+1));

}else {

System.out.println("用户:"+userInfo+",秒杀失败");

}

}else {

System.out.println("已经有20人秒杀成功,秒杀结束");

}

}catch (Exception e){

e.printStackTrace();

}finally {

jedis1.close();

}

});

}

executorService.shutdown();

}

效果如下图所示

06f1bce98451

1.png

如果有什么不对的地方请积极指出!

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Redis可以通过使用事务或Lua脚本来实现高并发下的抢购/秒杀功能。 1. 使用事务:抢购/秒杀的过程可以看做是一个先检查库存是否充足,再扣减库存的过程。使用Redis的事务可以保证这个过程是原子性的,即要么全部成功,要么全部失败。 具体实现方法如下: - 使用MULTI命令开启一个事务。 - 使用WATCH命令对库存进行监视(即设置监视器)。 - 使用GET命令获取当前库存。 - 判断库存是否充足,如果充足,则使用DECRBY命令扣减库存。 - 使用EXEC命令提交事务,如果提交成功,则说明扣减成功,否则说明被其他线程抢先扣减了库存。 示例代码如下: ```python def decrease_stock(redis_conn, stock_key): with redis_conn.pipeline() as pipeline: while True: try: pipeline.watch(stock_key) stock = int(pipeline.get(stock_key)) if stock > 0: pipeline.multi() pipeline.decr(stock_key) pipeline.execute() return True else: return False except WatchError: continue ``` 2. 使用Lua脚本:Lua脚本可以在Redis端原子性地执行多个命令,可以减少网络开销和锁竞争的问题。 具体实现方法如下: - 编写一个Lua脚本,该脚本首先使用GET命令获取当前库存,如果库存充足,则使用DECRBY命令扣减库存,否则返回0。 - 在Python中使用Redis的EVAL命令执行该Lua脚本。 示例代码如下: ```python def decrease_stock(redis_conn, stock_key): script = """ local stock = tonumber(redis.call('GET', KEYS[1])) if stock > 0 then redis.call('DECRBY', KEYS[1], 1) return 1 else return 0 end """ result = redis_conn.eval(script, 1, stock_key) return bool(result) ``` 以上两种方法都可以实现高并发下的抢购/秒杀功能,但是使用Lua脚本的方法效率更高,因为在Redis端执行命令可以减少网络开销和锁竞争的问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值