并发编程-简易秒杀实践(分布式锁+redis+springboot+redisson)

背景

分布式锁的核心是把并发操作强行串行化,在面试过程中非常常见,在高并发场景的编程下也非常常见。本wiki将进行简易的编程实践。
注:代码没有在线上工程中实际应用,仅供学习参考

重点关注: redis并不是最佳的实现分布式锁的方式,集群中master出现问题后,slave的数据并不能即时地同步锁信息。

实践过程

搭建环境

使用docker搭建redis环境

sudo docker run --name redis -p 6379:6379 -d redis

spring boot实现一个简易的web服务(seckill-demo)
代码参考:https://gitee.com/wangtonggui/spring-boot-demo.git
并发编程-简易秒杀实践(分布式锁+redis+springboot+redisson)_第1张图片
建议的秒杀场景
并发编程-简易秒杀实践(分布式锁+redis+springboot+redisson)_第2张图片
预先在redis中设置一个库存
set count 100
在这里插入图片描述

实现过程演进

并发编程-简易秒杀实践(分布式锁+redis+springboot+redisson)_第3张图片
如果上锁以后,web服务直接重启了,没有解锁,redis中的lock会一直存在

所以需要添加锁的过期时间(也就是自动解锁)。
并发编程-简易秒杀实践(分布式锁+redis+springboot+redisson)_第4张图片

问题:如果中间出现异常,而没有运行到delete锁的操作,怎么办?
使用try finally进行如下改进。
并发编程-简易秒杀实践(分布式锁+redis+springboot+redisson)_第5张图片

问题:如果自己创建的锁被别人删除了怎么办?
解决方法:每个锁都有一个唯一的编号
可以直接使用锁的value来进行存储,如下代码
并发编程-简易秒杀实践(分布式锁+redis+springboot+redisson)_第6张图片
问题:如果业务处理耗时过长(超过了锁的过期时间)怎么办?
解决方法:异步续命(即,在另外的一个线程,周期性的设置锁expiretime)
并发编程-简易秒杀实践(分布式锁+redis+springboot+redisson)_第7张图片
问题:所有的线程,都在一个锁(LOCK)上进行串行,明显是效率低的
解决方法:可以做成分段锁(类似于ConcurrentHashMap),每个锁分一些库存,提高并发效率。

redisson实现分布式锁

并发编程-简易秒杀实践(分布式锁+redis+springboot+redisson)_第8张图片

  • 实现uuid的存储与判断
  • 实现自动续期
  • 也可以实现分段锁

评价

redisson在大厂应用广泛,之后的wiki中进行实践。

你可能感兴趣的:(并发编程,#,redis面试)