首页 > 其他分享 >一个小小的乐观锁、悲观锁也能扯这么多

一个小小的乐观锁、悲观锁也能扯这么多

时间:2024-01-19 15:31:33浏览次数:36  
标签:小小的 库存 下单 更新 乐观 xx 加锁 悲观 stock


前言:我们一个普通的下单接口通常都包含如下三步操作,如果下单不成功的话将会返回给用户一个提示下单失败。

  1. 查询库存(select stock from xx where id = xx)
  2. 扣减更新库存(update xx set stock = stock - 1 where id = xx)
  3. 生成订单

如果是只有一个用户来请求下单接口,那么上述的操作毫无疑问是一定正常的,可是真实环境中可是处处存在并发的,如果此时库存数量为1,在高并发的情况下面用户一、二都查询到了库存,然后都来扣减库存那么会导致库存数量变为-1,这种情况是我们要杜绝的。

解决方案

  1. 悲观锁:我们可以在查询库存这一步加悲观锁(for update),然后常规的扣减库存操作。简而言之查询加锁,常规更新。悲观锁的体现:每次查询都认为有人来竞争资源,都会对资源进行加锁操作。在查库存阶段就已经对数据上锁
  2. 乐观锁 :查询库存不加锁,更新携带版本号更新,简而言之查询不加锁,更新加版本号。加锁的时机在更新库存的时候

悲观锁执行流程图(拿下单接口举例)

  1. 查询库存(select stock from xx where id = xx for update)
  2. 扣减更新库存(update xx set stock = stock - 1 where id = xx)
  3. 生成订单

下单接口伪代码

一个小小的乐观锁、悲观锁也能扯这么多_mysql

悲观锁下单接口实现流程

一个小小的乐观锁、悲观锁也能扯这么多_悲观锁_02

乐观锁执行流程(拿下单接口举例)

  1. 查询库存(select stock,version from xx where id = xx)
  2. 扣减更新库存(update xx set stock = stock - 1 ,version = version + 1 where id = xx and version = ?)
  3. 生成订单

乐观锁接口伪代码

一个小小的乐观锁、悲观锁也能扯这么多_数据库_03

一个小小的乐观锁、悲观锁也能扯这么多_数据库_04

悲观锁与乐观锁的使用场景以及优缺点分析

分析乐观锁与悲观锁的区别之前,我们要知道我们在执行

update xx set stock = stock-1 where id = 1

语句的时候,mysql会自动给id = 1的这条数据给上锁。而mysql进行上锁是会带来开销的。mysql上锁详解过程参考 mysq锁原理

悲观锁使用场景:适合并发冲突比较多的场景。

悲观锁缺点

  1. 无论有没有并发操作,一定会有互斥锁(for update)带来的开销。
  2. 使用不当还容易造成死锁的问题

乐观锁使用场景:适合并发冲突比较少的场景

乐观锁的优点:乐观锁衍生出来的目的就在于减少mysql中互斥锁(for update)的开销,在冲突比较少的情况下面对比悲观锁,乐观锁可以有效的减少mysql使用互斥锁给我们带来的开销,与其说乐观锁是一把锁,不如说乐观锁其实是一种我们能否修改数据的依据

什么叫做乐观锁适合并发冲突比较少的场景?

乐观锁是查询不加锁的,在有多个用户请求下单接口的时候,由于查询不加锁,那么这些用户都会来到更新库存这,而我们在执行一条update语句的时候会给所更新的记录加上一把行锁防止其他事务更新同一条数据。所以多个用户来到更新库存这里必定只有一个用户能更新成功(这里要理解并发与并行的区别)。而由于更新成功的用户进行了version++操作,等其他的用户接着更新的时候where条件根本就不会满足了,更新个锤子直接返回给用户下单失败的消息了,这就导致了一个问题,如果有100个人同时请求下单接口的话只有一个人会请求成功,其他的99个人直接就失败了。体验感不好啊

为什么说悲观锁适合并发冲突比较高的场景?

悲观锁在更新库存的时候,在查询库存的时候就已经对相应的记录给上过锁了,其次才是更新库存。假设有100个人同时请求下单接口,100个人会并发的去查库存,由于查库存是进行了加锁操作,只有一个人能查到库存,并且执行接下来的更新库存操作。剩下的99个人呢?由于竞争不到锁会卡在那里,直至等到锁释放才会正常的查询到库存然后执行更新库存的操作。虽然说这些人在等待锁释放的过程中会消耗一些时间,但是最起码不会直接给用户返回下单失败的信息吧。你敢说你在参与秒杀活动的时候,系统是秒回的嘛。


标签:小小的,库存,下单,更新,乐观,xx,加锁,悲观,stock
From: https://blog.51cto.com/u_16414043/9330462

相关文章

  • php redis 悲观锁
    悲观锁(PessimisticLock),顾名思义,就是每次处理redis数据都以最悲观的场景展开,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会block直到它拿到锁。传统的关系型数据库里边就用到了很多这种锁机制,比如行锁,表锁等,读锁,写锁等,都是在做......
  • 小小的日志,大大的坑
    1.背景压测过程中优化线程池以后单机qps存在性能瓶颈,优化过程中发现默认线程池及日志对性能存在严重的影响所以引发了一系列对日志优化的整理2.哪些场景可能导致性能问题在任何系统中,日志都是非常重要的组成部分,它是反映系统运行情况的重要依据,也是排查问题时的必要线索。绝大......
  • 乐观锁和悲观锁
    悲观锁认为⾃⼰在使⽤数据的时候⼀定有别的线程来修改数据,因此在获取数据的时候会先加锁,确保数据不会被别的线程修改。synchronized关键字和Lock的实现类都是悲观锁适合写操作多的场景,先加锁可以保证写操作时数据正确。调用方式//=============悲观锁的调⽤⽅式publicsynch......
  • 悲观锁、乐观锁、mybatis-plus实现乐观锁
    转载自:www.javaman.cn1、悲观锁、乐观锁乐观锁和悲观锁是两种用于处理并发操作的数据锁定策略。它们在处理多个事务尝试同时访问和修改同一数据时的方法有所不同。悲观锁(PessimisticLocking):概念:悲观锁是一种基于悲观态度的数据并发控制机制。它总是假设最坏的情况,即认为其他......
  • 悲观锁、乐观锁、mybatis-plus实现乐观锁
    悲观锁、乐观锁、mybatis-plus实现乐观锁转载自:www.javaman.cn1、悲观锁、乐观锁乐观锁和悲观锁是两种用于处理并发操作的数据锁定策略。它们在处理多个事务尝试同时访问和修改同一数据时的方法有所不同。悲观锁(PessimisticLocking):概念:悲观锁是一种基于悲观态度的数......
  • Django中实现事务的几种方式、事物的回滚和保存点、事务提交后,执行某个回调函数、Djan
    Django中实现事务的几种方式#https://zhuanlan.zhihu.com/p/622987268Django是支持事务操作的,它的默认事务行为是自动提交,具体表现形式为:每次数据库操作(比如调用save()方法)会立即被提交到数据库中。但是如果你希望把连续的SQL操作包裹在一个事务里,就需要手动开启事务#......
  • Self-Supervised Depth——一个小小的尝试
    一.选择原因。Self-SupervisedDepth是一种用于图像分割的深度学习模型。它是基于Self-Supervised模型的改进,通过引入链接模块和下采样模块来提高分割性能。Self-SupervisedDepth的核心思想是在Self-Supervised的基础上引入链接模块和下采样模块。链接模块用于捕捉不同尺度的特征......
  • 悲观剪枝
    https://www.bilibili.com/video/BV175411g7T7/?spm_id_from=333.337.search-card.all.click&vd_source=5a8099d424866bf1cdbfa5a7afadeca6个人理解,一个人腿被压了去医院,医生说在不做任何干预的情况下,有乐观(期望-1倍标准差),中性(期望),悲观(期望+1倍标准差)3种情况,乐观(休息2天就出院了......
  • 如何实现乐观锁
    乐观锁一般会使用版本号机制或CAS算法实现,CAS算法相对来说更多一些,这里需要格外注意。#版本号机制一般是在数据表中加上一个数据版本号 version 字段,表示数据被修改的次数。当数据被修改时,version 值会加一。当线程A要更新数据值时,在读取数据的同时也会读取 version 值,在......
  • 鼻咽癌的一些感悟!(乐观坚韧是最难得)
    1.记得之前的自己从小就害怕进医院,怕手术,更怕癌症!所以努力健康作息,不抽烟,少酒(因为聚餐职场避免不了),不吃外卖,不吃奶茶冰淇淋等,晚上空余下班时间散散步,跳绳,在这之前总觉得自己是天选之子,没生过什么病,感冒,新冠抗几天就康复了,感觉这辈子都会顺顺利利的,不会和大......