一、分布式锁应用场景:
    1.互联网秒杀
    2.抢优惠卷
    3.接口幂等性校验
    4.其它一些场景

    二、几大最常见的秒杀减库存的例子与问题分析:
    1.用synchronized锁

    1. //方式1:用synchronized锁控制并发
    2. @RequestMapping("/deduct_stock1")
    3. public String deductStock1() {
    4. String lockKey = "product_101";
    5. synchronized (this){
    6. int count = Integer.parseInt(stringRedisTemplate.opsForValue().get(lockKey));
    7. if(count<=0){
    8. return "库存为0";
    9. }
    10. Boolean result = stringRedisTemplate.opsForValue().setIfAbsent(lockKey, "azhi");
    11. if (!result) {
    12. return "error_code";
    13. }
    14. }
    15. return "end";
    16. }

    ** 问题:如果部署是在单机tomcat情况下,这个代码是没有问题的,synchronized确保了在多线程情况下同时只会有一个线程执行到代码块里的代码,但如果是分布式部署,那在高并发场景下就不可控了

    2.用redis分布式锁

    1. //方式2:用redis分布式锁,拿到锁的才能操作,操作完再释放锁给后面的人去拿锁
    2. @RequestMapping("/deduct_stock2")
    3. public String deductStock2() {
    4. String lockKey = "product_101";
    5. try {
    6. //jedis.setnx(k,v)
    7. Boolean result = stringRedisTemplate.opsForValue().setIfAbsent(lockKey, "azhi");
    8. if (!result) {
    9. return "error_code";
    10. }
    11. //这里执行其它的一些逻辑。。。
    12. } finally {
    13. stringRedisTemplate.delete(lockKey);
    14. }
    15. return "end";
    16. }

    ** 问题:如果在拿到锁并在执行其它逻辑过程中死机了,代码没执行到finally,锁没解到,那么锁就一直在,后面的人会一直拿不到锁

    3.用redis分布式锁,并给锁加一个过期时间

    1. //方式3:用redis分布式锁,并给锁加一个过期时间
    2. @RequestMapping("/deduct_stock3")
    3. public String deductStock3() {
    4. String lockKey = "product_101";
    5. try {
    6. //stringRedisTemplate.expire(lockKey, 10, TimeUnit.SECONDS);
    7. Boolean result = stringRedisTemplate.opsForValue().setIfAbsent(lockKey, "azhi", 10, TimeUnit.SECONDS);
    8. if (!result) {
    9. return "error_code";
    10. }
    11. //这里执行其它的一些逻辑。。。
    12. } finally {
    13. stringRedisTemplate.delete(lockKey);
    14. }
    15. return "end";
    16. }

    ** 问题:如在执行其它逻辑的代码块用时过多,锁的过期时间到了,锁自动失效了,其它线程成功拿到了锁,但最后finally里却把其它用户加的锁给删除了。

    加入clientId实现谁加的锁谁来删除

    1. //方式3的优化:加入clientId来防止最后finally里不小心把其它用户加的锁给删除的问题
    2. @RequestMapping("/deduct_stock33")
    3. public String deductStock33() {
    4. String lockKey = "product_101";
    5. //这里生成一个uuid,记录锁的生成者。
    6. String clientId = UUID.randomUUID().toString();
    7. try {
    8. //stringRedisTemplate.expire(lockKey, 10, TimeUnit.SECONDS);
    9. Boolean result = stringRedisTemplate.opsForValue().setIfAbsent(lockKey, clientId, 10, TimeUnit.SECONDS);
    10. if (!result) {
    11. return "error_code";
    12. }
    13. //这里执行其它的一些逻辑。。。
    14. } finally {
    15. //这里判断锁是不是由自己这个线程加的,如果不是的话就不删除锁,
    16. if (clientId.equals(stringRedisTemplate.opsForValue().get(lockKey))) {
    17. stringRedisTemplate.delete(lockKey);
    18. }
    19. }
    20. return "end";
    21. }

    问题:如果逻辑处理代码确实要处理15秒,那么锁过期之后其它用户还是一样能拿到锁,这样依然是会有bug存在,如超卖之类的问题。也就是锁过期时间我们并不好根据逻辑代码去控制。
    优化思路1:可以在主线程中加锁,然后开启一个分线程去判断给锁续命, 如果锁到期了依然还没执行到finally的话给锁续时间。这个思路既可以解决方法1的问题,又可以解决到方法2中的问题,但实现起来会比较麻烦,坑多,推荐用redisson
    ** 优化思路2:用lua脚本实现

    4.用lua脚本去实现

    1. //方式4,用lua脚本实现
    2. @RequestMapping("/deduct_stock4")
    3. public String deductStock4
    4. {
    5. String script = " local count = redis.call('get', KEYS[1]) " +
    6. " local a = tonumber(count) " +
    7. " local b = tonumber(ARGV[1]) " +
    8. " if a >= b then " +
    9. " redis.call('set', KEYS[1], a-b) " +
    10. " return 1 " +
    11. " end " +
    12. " return 0 ";
    13. //商品product_101,减10个库存
    14. Object obj = jedis.eval(script, Arrays.asList("product_101"), Arrays.asList("10"));
    15. }

    redis不仅仅可以执行命令,也可以执行带逻辑运算的lua脚本,在lua脚本里的运算都是带原子性的,要么就全部不执行要么就全部执行,一般也是用lua脚本去代替redis的事务。而且在高并发情况下多个提交上去的lua脚本都会在redis队列里排队等待被执行的,也就不存在并发时数据超卖的问题了。

    5.用redisson分布式锁实现

    1. @Autowired
    2. private RedissonClient redisson;
    3. @Autowired
    4. private StringRedisTemplate stringRedisTemplate;
    5. /***
    6. * 方式5:用redisson
    7. * @return
    8. */
    9. @RequestMapping("/deduct_stock5")
    10. public String deductStock5() {
    11. String lockKey = "product_101";
    12. RLock redissonLock = redisson.getLock(lockKey);
    13. try {
    14. //加锁,实现锁续命功能,默认是30秒过期,每过10会进行一次判断 setIfAbsent(lockKey, clientId, 30, TimeUnit.SECONDS);
    15. redissonLock.lock();
    16. int stock = Integer.parseInt(stringRedisTemplate.opsForValue().get("stock")); // jedis.get("stock")
    17. if (stock > 0) {
    18. int realStock = stock - 1;
    19. stringRedisTemplate.opsForValue().set("stock", realStock + ""); // jedis.set(key,value)
    20. System.out.println("扣减成功,剩余库存:" + realStock);
    21. } else {
    22. System.out.println("扣减失败,库存不足");
    23. }
    24. } finally {
    25. redissonLock.unlock();
    26. }
    27. return "end";
    28. }

    redisson引用与配置
    引用了redisson-spring-boot-starter就不需要再引用spring-boot-starter-data-redis了,如果用jedis的话那需要另外引入jedis的依赖

    1. <dependencies>
    2. <dependency>
    3. <groupId>org.springframework.boot</groupId>
    4. <artifactId>spring-boot-starter-web</artifactId>
    5. <version>2.3.9.RELEASE</version>
    6. </dependency>
    7. <dependency>
    8. <groupId>org.redisson</groupId>
    9. <artifactId>redisson-spring-boot-starter</artifactId>
    10. <version>3.13.6</version>
    11. </dependency>
    12. </dependencies>
    13. <!--打包jar-->
    14. <build>
    15. <finalName>test2</finalName>
    16. <plugins>
    17. <!--spring-boot-maven-plugin-->
    18. <plugin>
    19. <groupId>org.springframework.boot</groupId>
    20. <artifactId>spring-boot-maven-plugin</artifactId>
    21. <version>2.4.2</version>
    22. <!--解决打包出来的jar文件中没有主清单属性问题-->
    23. <executions>
    24. <execution>
    25. <goals>
    26. <goal>repackage</goal>
    27. </goals>
    28. </execution>
    29. </executions>
    30. </plugin>
    31. <plugin>
    32. <groupId>org.apache.maven.plugins</groupId>
    33. <artifactId>maven-compiler-plugin</artifactId>
    34. <configuration>
    35. <source>8</source>
    36. <target>8</target>
    37. </configuration>
    38. </plugin>
    39. </plugins>
    40. </build>

    单机模式和集群模式配置类:

    1. @Configuration
    2. public class MyRedissonConfig {
    3. @Bean(destroyMethod="shutdown")
    4. RedissonClient redisson() throws IOException {
    5. //1、单机模式
    6. Config config = new Config();
    7. config.useSingleServer()
    8. .setAddress("redis://xxxx:6379").setPassword("xxxx").setDatabase(1);
    9. //2、集群模式
    10. /*config.useClusterServers()
    11. .addNodeAddress("").setPassword("")
    12. .addNodeAddress("").setPassword("")
    13. .addNodeAddress("").setPassword("")
    14. .addNodeAddress("").setPassword("")
    15. .addNodeAddress("").setPassword("")
    16. .addNodeAddress("").setPassword("");*/
    17. return Redisson.create(config);
    18. }
    19. }

    如下,并发请求下不会出现超卖情况。
    image.png

    6.redisson实现分布式锁原理
    image.png
    7.redis分段锁
    场景:商品IP12准备搞促销,redis采用的是cluster集群,IP12分配到集群节点A上,库存1000个,需要并发性能提升10倍左右。
    解决方案:可以通过设置商品key的hash值让1000个库存平均分配到10个不同的redis集群节点上。

    8.Redis分布式锁与Zookeeper分布式锁区别
    上面这个场景其实还可以用zookeeper分布式锁来实现,zookeeper的数据也是写在内存里,而且zookeeper分布式锁是强一致性的,zookeeper有写数据过半机制,也就是一个写数据操作要过半机器都写入成功才会给客户端返回成功,从而保证了数据的强一致性,当然性能就没有Redis那么高了,毕竟每一个写操作都要走过半机制是很影响性能的。
    而Redis没有写数据过半机制,只要主节点写成功了就会返回成功给客户端,从节点的数据同步是异步进行处理的,如果主节点写入成功,而从节点还没同步到数据的情况下主节点掉线了redis选举了一个从节点当做主节点了,那么就会出现锁丢失数据不一致的情况。但Redis提供了配置min-replicas-to-write N参数,让master停止写入的方式,当健康的slave的个数小于N,mater就禁止写入,这个配置虽然不能保证N个slave都一定能接收到master的写操作,但是能避免没有足够健康的slave的时候,master拒绝写入只能读来尽可能避免数据的丢失,但是这样也没有办法百分百保证数据的一致。
    所以对于分布式锁这个场景到底是要选择Redis还是选择Zookeeper呢?
    这个问题就要看你是如何进行取舍的了,如果不是很注重高可用,一定要百分百的一致性的话那就用Zookeeper,如果觉得更加注重系统的高可用,小概率的不一致可以接受的话那就用Redis。