热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

开发笔记:面试官问:说说悲观锁乐观锁分布式锁?都在什么场景下使用?有什么技巧?

篇首语:本文由编程笔记#小编为大家整理,主要介绍了面试官问:说说悲观锁乐观锁分布式锁?都在什么场景下使用?有什么技巧?相关的知识,希望对你有一定的参考价值。

篇首语:本文由编程笔记#小编为大家整理,主要介绍了面试官问:说说悲观锁乐观锁分布式锁?都在什么场景下使用?有什么技巧?相关的知识,希望对你有一定的参考价值。







来源 | www.cnblogs.com/jackyfei/p/12142840.html


如何确保一个方法,或者一块代码在高并发情况下,同一时间只能被一个线程执行,单体应用可以使用并发处理相关的 API 进行控制,但单体应用架构演变为分布式微服务架构后,跨进程的实例部署,显然就没办法通过应用层锁的机制来控制并发了。


那么锁都有哪些类型,为什么要使用锁,锁的使用场景有哪些?


锁类别


不同的应用场景对锁的要求各不相同,我们先来看下锁都有哪些类别,这些锁之间有什么区别。


  • 悲观锁(synchronize)

    • Java 中的重量级锁 synchronize

    • 数据库行锁


  • 乐观锁

    • Java 中的轻量级锁 volatile 和 CAS

    • 数据库版本号


  • 分布式锁(Redis锁)


乐观锁


就好比说是你是一个生活态度乐观积极向上的人,总是往最好的情况去想,比如你每次去获取共享数据的时候会认为别人不会修改,所以不会上锁,但是在更新的时候你会判断这期间有没有人去更新这个数据。


乐观锁使用在前,判断在后。我们看下伪代码:


reduce()
{
    select total_amount from table_1
    if(total_amount < amount ){
          return failed.  
    }  
    //其他业务逻辑
    update total_amount &#61; total_amount - amount where total_amount > amount; }

  • 数据库的版本号属于乐观锁&#xff1b;

  • 通过CAS算法实现的类属于乐观锁。


悲观锁


悲观锁是怎么理解呢&#xff1f;相对乐观锁刚好反过来&#xff0c;总是假设最坏的情况&#xff0c;假设你每次拿数据的时候会被其他人修改&#xff0c;所以你在每次共享数据的时候会对他加一把锁&#xff0c;等你使用完了再释放锁&#xff0c;再给别人使用数据。


悲观锁判断在前&#xff0c;使用在后。我们也看下伪代码&#xff1a;


reduce()
{
    //其他业务逻辑
    int num &#61; update total_amount &#61; total_amount - amount where total_amount > amount; 
   if(num &#61;&#61;1 ){
          //业务逻辑.  
    } 
}

  • Java中的的synchronize是重量级锁 &#xff0c;属于悲观锁&#xff1b;

  • 数据库行锁属于悲观锁&#xff1b;


扣减操作案例


这里举一个非常常见的例子&#xff0c;在高并发情况下余额扣减&#xff0c;或者类似商品库存扣减&#xff0c;也可以是资金账户的余额扣减。扣减操作会发生什么问题呢&#xff1f;很容易可以看到&#xff0c;可能会发生的问题是扣减导致的超卖&#xff0c;也就是扣减成了负数。


举个例子&#xff0c;比如我的库存数据只有100个。并发情况下第1笔请求卖出100个&#xff0c;第2批卖出100元&#xff0c;导致当前的库存数量为负数。遇到这种场景应该如何破解呢&#xff1f;这里列举四种方案。


方案1&#xff1a;同步排它锁


这时候很容易想到最简单的方案&#xff1a;同步排它锁(synchronize)。但是排他锁的缺点很明显&#xff1a;


  • 其中一个缺点是&#xff0c;线程串行导致的性能问题&#xff0c;性能消耗比较大。

  • 另一个缺点是无法解决分布式部署情况下跨进程问题&#xff1b;


方案2&#xff1a;数据库行锁


第二我们可能会想到&#xff0c;那用数据库行锁来锁住这条数据&#xff0c;这种方案相比排它锁解决了跨进程的问题&#xff0c;但是依然有缺点。


  • 其中一个缺点就是性能问题&#xff0c;在数据库层面会一直阻塞&#xff0c;直到事务提交&#xff0c;这里也是串行执行&#xff1b;

  • 第二个需要注意设置事务的隔离级别是Read Committed&#xff0c;否则并发情况下&#xff0c;另外的事务无法看到提交的数据&#xff0c;依然会导致超卖问题&#xff1b;

  • 缺点三是容易打满数据库连接&#xff0c;如果事务中有第三方接口交互(存在超时的可能性)&#xff0c;会导致这个事务的连接一直阻塞&#xff0c;打满数据库连接。

  • 最后一个缺点&#xff0c;容易产生交叉死锁&#xff0c;如果多个业务的加锁控制不好&#xff0c;就会发生AB两条记录的交叉死锁。


方案3&#xff1a;redis分布式锁


前面的方案本质上是把数据库当作分布式锁来使用&#xff0c;所以同样的道理&#xff0c;redis&#xff0c;zookeeper都相当于数据库的一种锁&#xff0c;其实当遇到加锁问题&#xff0c;代码本身无论是synchronize或者各种lock使用起来都比较复杂&#xff0c;所以思路是把代码处理一致性的问难题交给一个能够帮助你处理一致性的问题的专业组件&#xff0c;比如数据库&#xff0c;比如redis&#xff0c;比如zookeeper等。


这里我们分析下分布式锁的优缺点&#xff1a;


  • 优点&#xff1a;

    • 可以避免大量对数据库排他锁的征用&#xff0c;提高系统的响应能力&#xff1b;


  • 缺点&#xff1a;

    • 设置锁和设置超时时间的原子性&#xff1b;

    • 不设置超时时间的缺点&#xff1b;

    • 服务宕机或线程阻塞超时的情况&#xff1b;

    • 超时时间设置不合理的情况&#xff1b;



加锁和过期设置的原子性


redis加锁的命令setnx&#xff0c;设置锁的过期时间是expire&#xff0c;解锁的命令是del&#xff0c;但是2.6.12之前的版本中&#xff0c;加锁和设置锁过期命令是两个操作&#xff0c;不具备原子性。如果setnx设置完key-value之后&#xff0c;还没有来得及使用expire来设置过期时间&#xff0c;当前线程挂掉了或者线程阻塞&#xff0c;会导致当前线程设置的key一直有效&#xff0c;后续的线程无法正常使用setnx获取锁&#xff0c;导致死锁。


针对这个问题&#xff0c;redis2.6.12以上的版本增加了可选的参数&#xff0c;可以在加锁的同时设置key的过期时间&#xff0c;保证了加锁和过期操作原子性的。


但是&#xff0c;即使解决了原子性的问题&#xff0c;业务上同样会遇到一些极端的问题&#xff0c;比如分布式环境下&#xff0c;A获取到了锁之后&#xff0c;因为线程A的业务代码耗时过长&#xff0c;导致锁的超时时间&#xff0c;锁自动失效。后续线程B就意外的持有了锁&#xff0c;之后线程A再次恢复执行&#xff0c;直接用del命令释放锁&#xff0c;这样就错误的将线程B同样Key的锁误删除了。代码耗时过长还是比较常见的场景&#xff0c;假如你的代码中有外部通讯接口调用&#xff0c;就容易产生这样的场景。


设置合理的时长


刚才讲到的线程超时阻塞的情况&#xff0c;那么如果不设置时长呢&#xff0c;当然也不行&#xff0c;如果线程持有锁的过程中突然服务宕机了&#xff0c;这样锁就永远无法失效了。同样的也存在锁超时时间设置是否合理的问题&#xff0c;如果设置所持有时间过长会影响性能&#xff0c;如果设置时间过短&#xff0c;有可能业务阻塞没有处理完成&#xff0c;是否可以合理的设置锁的时间?


续命锁


这是一个很不容易解决的问题&#xff0c;不过有一个办法能解决这个问题&#xff0c;那就是续命锁&#xff0c;我们可以先给锁设置一个超时时间&#xff0c;然后启动一个守护线程&#xff0c;让守护线程在一段时间之后重新去设置这个锁的超时时间&#xff0c;续命锁的实现过程就是写一个守护线程&#xff0c;然后去判断对象锁的情况&#xff0c;快失效的时候&#xff0c;再次进行重新加锁&#xff0c;但是一定要判断锁的对象是同一个&#xff0c;不能乱续。


同样&#xff0c;主线程业务执行完了&#xff0c;守护线程也需要销毁&#xff0c;避免资源浪费&#xff0c;使用续命锁的方案相对比较而言更复杂&#xff0c;所以如果业务比较简单&#xff0c;可以根据经验类比&#xff0c;合理的设置锁的超时时间就行。


方案4&#xff1a;数据库乐观锁


数据库乐观锁加锁的一个原则就是尽量想办法减少锁的范围。锁的范围越大&#xff0c;性能越差&#xff0c;数据库的锁就是把锁的范围减小到了最小。我们看下面的伪代码


reduce()
{
    select total_amount from table_1
    if(total_amount < amount ){
          return failed.  
    }  
    //其他业务逻辑
    update total_amount &#61; total_amount - amount;  
}

我们可以看到修改前的代码是没有where条件的。修改后&#xff0c;再加where条件判断&#xff1a;总库存大于将被扣减的库存。


update total_amount &#61; total_amount - amount where total_amount > amount

如果更新条数返回0&#xff0c;说明在执行过程中被其他线程抢先执行扣减&#xff0c;并且避免了扣减为负数。


但是这种方案还会涉及一个问题&#xff0c;如果在之前的update代码中&#xff0c;以及其他的业务逻辑中还有一些其他的数据库写操作的话&#xff0c;那这部分数据如何回滚呢&#xff1f;


我的建议是这样的&#xff0c;你可以选择下面这两种写法&#xff1a;


  • 利用事务回滚写法&#xff1a;


我们先给业务方法增加事务&#xff0c;方法在扣减库存影响条数为零的时候扔出一个异常&#xff0c;这样对他之前的业务代码也会回滚。


reduce()
{
    select total_amount from table_1
    if(total_amount < amount ){
          return failed.  
    }  
    //其他业务逻辑
    int num &#61; update total_amount &#61; total_amount - amount where total_amount > amount;   if(num&#61;&#61;0) throw Exception;}

  • 第二种写法


reduce()
{
    //其他业务逻辑
    int num &#61; update total_amount &#61; total_amount - amount where total_amount > amount;    if(num &#61;&#61;1 ){
          //业务逻辑.  
    }  else{    throw Exception;  }
}

首先执行update业务逻辑&#xff0c;如果执行成功了再去执行逻辑操作&#xff0c;这种方案是我相对比较建议的方案。在并发情况下对共享资源扣减操作可以使用这种方法&#xff0c;但是这里需要引出一个问题&#xff0c;比如说万一其他业务逻辑中的业务&#xff0c;因为特殊原因失败了该怎么办呢&#xff1f;比如说在扣减过程中服务OOM了怎么办&#xff1f;


我只能说这些非常极端的情况&#xff0c;比如突然宕机中间数据都丢了&#xff0c;这种极少数的情况下只能人工介入&#xff0c;如果所有的极端情况都考虑到&#xff0c;也不现实。我们讨论的重点是并发情况下&#xff0c;共享资源的操作如何加锁的问题。


总结


最后我来给你总结一下&#xff0c;如果你可以非常熟练的解决这类问题&#xff0c;第一时间肯定想到的是&#xff1a;数据库版本号解决方案或者分布式锁的解决方案&#xff1b;但是如果你是一个初学者&#xff0c;相信你一定会第一时间考虑到Java中提供的同步锁或者数据库行锁。


END


免费领取 1000&#43; 道面试资料&#xff01;&#xff01;小编这里有一份面试宝典《Java 核心知识点.pdf》&#xff0c;覆盖了 JVM&#xff0c;锁、高并发、Spring原理、微服务、数据库、Zookeep人、数据结构等等知识点&#xff0c;包含 Java 后端知识点 1000&#43; 个&#xff0c;部分如下&#xff1a;




如何获取&#xff1f;加小编微信&#xff0c;回复【1024】






推荐阅读
  • 基于分布式锁的防止重复请求解决方案
    一、前言关于重复请求,指的是我们服务端接收到很短的时间内的多个相同内容的重复请求。而这样的重复请求如果是幂等的(每次请求的结果都相同,如查 ... [详细]
  • 深入理解Java虚拟机的并发编程与性能优化
    本文主要介绍了Java内存模型与线程的相关概念,探讨了并发编程在服务端应用中的重要性。同时,介绍了Java语言和虚拟机提供的工具,帮助开发人员处理并发方面的问题,提高程序的并发能力和性能优化。文章指出,充分利用计算机处理器的能力和协调线程之间的并发操作是提高服务端程序性能的关键。 ... [详细]
  • 一次上线事故,30岁+的程序员踩坑经验之谈
    本文主要介绍了一位30岁+的程序员在一次上线事故中踩坑的经验之谈。文章提到了在双十一活动期间,作为一个在线医疗项目,他们进行了优惠折扣活动的升级改造。然而,在上线前的最后一天,由于大量数据请求,导致部分接口出现问题。作者通过部署两台opentsdb来解决问题,但读数据的opentsdb仍然经常假死。作者只能查询最近24小时的数据。这次事故给他带来了很多教训和经验。 ... [详细]
  • 本文整理了Java中java.lang.NoSuchMethodError.getMessage()方法的一些代码示例,展示了NoSuchMethodErr ... [详细]
  • linux进阶50——无锁CAS
    1.概念比较并交换(compareandswap,CAS),是原⼦操作的⼀种,可⽤于在多线程编程中实现不被打断的数据交换操作࿰ ... [详细]
  • 本文由编程笔记#小编为大家整理,主要介绍了源码分析--ConcurrentHashMap与HashTable(JDK1.8)相关的知识,希望对你有一定的参考价值。  Concu ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • ALTERTABLE通过更改、添加、除去列和约束,或者通过启用或禁用约束和触发器来更改表的定义。语法ALTERTABLEtable{[ALTERCOLUMNcolu ... [详细]
  • CentOS 7部署KVM虚拟化环境之一架构介绍
    本文介绍了CentOS 7部署KVM虚拟化环境的架构,详细解释了虚拟化技术的概念和原理,包括全虚拟化和半虚拟化。同时介绍了虚拟机的概念和虚拟化软件的作用。 ... [详细]
  • 单点登录原理及实现方案详解
    本文详细介绍了单点登录的原理及实现方案,其中包括共享Session的方式,以及基于Redis的Session共享方案。同时,还分享了作者在应用环境中所遇到的问题和经验,希望对读者有所帮助。 ... [详细]
  • 深入理解Kafka服务端请求队列中请求的处理
    本文深入分析了Kafka服务端请求队列中请求的处理过程,详细介绍了请求的封装和放入请求队列的过程,以及处理请求的线程池的创建和容量设置。通过场景分析、图示说明和源码分析,帮助读者更好地理解Kafka服务端的工作原理。 ... [详细]
  • 深度学习中的Vision Transformer (ViT)详解
    本文详细介绍了深度学习中的Vision Transformer (ViT)方法。首先介绍了相关工作和ViT的基本原理,包括图像块嵌入、可学习的嵌入、位置嵌入和Transformer编码器等。接着讨论了ViT的张量维度变化、归纳偏置与混合架构、微调及更高分辨率等方面。最后给出了实验结果和相关代码的链接。本文的研究表明,对于CV任务,直接应用纯Transformer架构于图像块序列是可行的,无需依赖于卷积网络。 ... [详细]
  • MPLS VP恩 后门链路shamlink实验及配置步骤
    本文介绍了MPLS VP恩 后门链路shamlink的实验步骤及配置过程,包括拓扑、CE1、PE1、P1、P2、PE2和CE2的配置。详细讲解了shamlink实验的目的和操作步骤,帮助读者理解和实践该技术。 ... [详细]
  • JVM:33 如何查看JVM的Full GC日志
    1.示例代码packagecom.webcode;publicclassDemo4{publicstaticvoidmain(String[]args){byte[]arr ... [详细]
  • 初识java关于JDK、JRE、JVM 了解一下 ... [详细]
author-avatar
无正道
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有