浅谈Java分布式架构下如何实现分布式锁

 更新时间:2020年07月24日 16:42:46   作者:林口  
这篇文章主要介绍了浅谈Java分布式架构下如何实现分布式锁,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

  01分布式锁运用场景

  互联网秒杀,抢优惠卷,接口幂等性校验。咱们以互联网秒杀为例。

 @RestController
  @Slf4j
  publicclassIndexController{
  @Autowired
  privateRedissonredission;
  @Autowired
  privateStringRedisTemplatestringRedisTemplate;
  @RequestMapping("/deduct_stock")
  publicStringdeductStock()throwsInterruptedException{
  intstock=Integer.parseInt
  (stringRedisTemplate.opsForValue().get("stock"));
  if(stock>0){
  intrealStock=stock-1;
  stringRedisTemplate.opsForValue().set("stock",realStock+"");
  log.info("扣减成功,剩下库存+"+realStock);
  }else{
  return"库存不足";
  }
  return"end";
  }
  }

  上面代码假定多个线程一起进来,读到胡成果相同,写入的成果也相同。造成多个减库存操作,只减少了一次。

 好多人想到的处理方式是加synchronized,但是假如是集群项目呢?搭建的是负载均衡项目。打开了两个jvm进程,jvm进程锁失效。

  用nginx做负载均衡,启用两个服务。进行下压测。成果和咱们想的相同,用suynchronized关键字加锁,两台服务器打印的剩下库存有相同的。所以这种方法并没有完成线程安全。

  02根据redisson完成分布式锁

  咱们redis中有一个SETNXKEYVALUE,这个指令的时刻复杂度为O(1),只在键key不存在的情况下,将键key的值设置为value,若键key现已存在,则setnx指令不做任何操作,指令在设置成功时返回一,失利时分返回0.

  那么咱们能够简单用这个指令,完成一个十分简单的分布式锁。Springboot结构中咱们运用这个:

  stringRedisTemplate.opsForValue().setIfAbsent(),它底层是对setnx的封装。

  这样完成有什么问题?

  假如第一个线程抛了异常中断了,其它线程就永远不能处理了,所以咱们加上trycatch,finally。在finally里面delete咱们存的key值。

  假如程序宕机怎么办?不是可捕获异常怎么办?

  这时分程序履行到没有释放锁,即key值一向存在,在发动web服务的时分其它线程也一向不等待。这时分会一向存在死锁,咱们是不是参加锁的超时时刻是否能够,即
  stringRedisTemplate.expire()

  程序迭代的过程是逐步的优化,假定我还没来的及给锁设置时长,服务就宕掉了,这时分该怎么办。

  用两行指令履行咱们能够换成一行的指令,及设置key值的时分,一起设置时长,stringRedisTemplate.opsForValue().setIfAbsent()。合并成一个原子操作。

  这样写的化问题还是有的,假如你设置超时时刻为10秒,而我履行事务代码为小于10秒,还没履行完,就把锁释放掉了。高并发场景下你不能决议履行顺序.

  能够不能够自己加的锁,自己删去,别的线程不能进行操作?
  咱们给每个线程设置一个uuid,唯一id。这样轻轻松松解决了。
  商场上面现已有许多许多成熟的结构,咱们能够不需要去手写分布式结构来完成。

到此这篇关于浅谈Java分布式架构下如何实现分布式锁的文章就介绍到这了,更多相关Java分布式架构  分布式锁内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 浅谈Java中spring 线程异步执行

    浅谈Java中spring 线程异步执行

    这篇文章主要介绍了浅谈spring 线程异步执行,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-07-07
  • java组件commons-fileupload实现文件上传、下载、在线打开

    java组件commons-fileupload实现文件上传、下载、在线打开

    这篇文章主要介绍了java组件commons-fileupload实现文件上传、下载、在线打开,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-10-10
  • Java数据结构之最小堆和最大堆的原理及实现详解

    Java数据结构之最小堆和最大堆的原理及实现详解

    在计算机科学中,堆(heap) 的实现是一种基于树的特殊的数据结构,它可以在数组上构建出树的结构体,并满足堆的属性。本文就来和大家详细聊聊Java数据结构中的堆,感兴趣的可以了解一下
    2022-09-09
  • SpringAop @Around执行两次的原因及解决

    SpringAop @Around执行两次的原因及解决

    这篇文章主要介绍了SpringAop @Around执行两次的原因及解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-07-07
  • Mybatis-Plus如何使用分页实例详解

    Mybatis-Plus如何使用分页实例详解

    最近在研究mybatis,然后就去找简化mybatis开发的工具,下面这篇文章主要给大家介绍了关于Mybatis-Plus如何使用分页的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-03-03
  • 关于IDEA MybatisX插件的使用小技巧

    关于IDEA MybatisX插件的使用小技巧

    这篇文章主要介绍了关于MybatisX插件的使用小技巧,MybatisX是IDEA的一款第三方插件,可以极大地提升我们的开发效率,一起来看看吧
    2023-03-03
  • 探究Android系统中解析JSON数据的方式

    探究Android系统中解析JSON数据的方式

    这篇文章主要介绍了探究Android系统中解析JSON数据的方式,文中讲到了使用Java代码实现的处理JSON的一些主要方法,需要的朋友可以参考下
    2015-07-07
  • Java中队列Queue和Deque的区别与代码实例

    Java中队列Queue和Deque的区别与代码实例

    学过数据结构的,一定对队列不陌生,java也实现了队列,下面这篇文章主要给大家介绍了关于Java中队列Queue和Deque区别的相关资料,需要的朋友可以参考下
    2021-08-08
  • Spring Cloud调用Ribbon的步骤

    Spring Cloud调用Ribbon的步骤

    Ribbon是Netflix发布的开源项目,主要功能是提供客户端的软件负载均衡算法和服务调用。本文将讲述Spring Cloud调用Ribbon的方法
    2021-05-05
  • Java使用Apache POI操作Excel详解

    Java使用Apache POI操作Excel详解

    在Java中操作Excel是日常工作中经常遇到的问题,而Apache Poi是一种流行且广泛使用的方式,它提供了各种库和工具,所以本文就来详细如何使用Apache Poi来进行Excel文件操作吧
    2023-06-06

最新评论