浅谈实现数据库乐观锁和悲观锁
目录
- 数据库乐观锁
- 适用场景
- 定义
- 实现方式
- 优点与缺点分析
- 优点
- 缺点
- 案例
- 数据库悲观锁
- 适用场景
- 定义
- 实现方式
- 悲观锁优缺点分析
- 优点
- 缺点
在单实例JVM中,常见的处理并发问题的方法有很多,比如synchronized关键字进行访问控制、volatile关键字、ReentrantLock等常用方法。但是在分布式环境中,上述方法却不能在跨jvm场景中用于处理并发问题,当业务场景需要对分布式环境中的并发问题进行处理时,需要使用其他方式来实现,如数据库锁机制、缓存数据库如redis以及zookeeper分布式锁等。
本文主要介绍数据库中常用的乐观锁和悲观锁的实现以及优缺点。
数据库乐观锁
适用场景
乐观锁比较适合并发量不高,并且写操作不频繁的场景
定义
系统认为数据的更新在大多数情况下是不会产生冲突的,只在数据库更新操作提交的时候才对数据作冲突检测。如果检测的结果出现了与预期数据不一致的情况,则返回失败信息。
实现方式
- 借助数据库表增加一个版本号的字段version(数字类型),每次更新一行记录,都使得该行版本号加一,开始更新之前先获取version的值,更新提交的时候带上之前获取的version值与当前version值作比较,如果不相等则说明version值发生了变化则检测到了并发冲突,本次操作执行失败,如果相等则操作执行成功。
update table set columnA = 1,version=version+1 where id=#{id} and version = #{oldVersion}
借助行更新时间时间戳,检测方法则与方式1相似,即更新操作执行前先获取记录当前的更新时间,在提交更新时,检测当前更新时间是否与更新开始时获取的更新时间时间戳相等。
前面2种方式都是提交的时候检测版本有没有改变,只要有变化都会失败,而有一类场景当字段只需要满足一个区间范围并不关心是否有数据更新冲突,且本身进行更新并且作为判断条件时,可不借助其他字段,对字段本身作判断即可。例如一个较常见的场景:库存的扣减,只要扣减后的值大于等于零即可。
update product set rest = rest– #{deduct} where name = ‘abc’ and rest >= #{deduct}
优点与缺点分析
优点
由于在检测数据冲突时并不依赖数据库本身的锁机制,不会影响请求的性能,当产生并发且并发量较小的时候只有少部分请求会失败。
缺点
需要对表的设计增加额外的字段,增加了数据库的冗余,另外,当应用并发量高的时候,version值在频繁变化,则会导致大量请求失败,影响系统的可用性。我们通过上述sql语句还可以看到,数据库锁都是作用于同一行数据记录上,这就导致一个明显的缺点,在一些特殊场景,如大促、秒杀等活动开展的时候,大量的请求同时请求同一条记录的行锁,会对数据库产生很大的写压力。
案例
mysql增加version字段实现乐观锁,实现高并发下的订单库存的并发控制,通过开启多线程同时处理模拟多个请求同时到达的情况 + 同一事务中使用多个乐观锁的情况处理
http://www.cnblogs.com/sxdcgaq8080/p/9454161.html
数据库悲观锁
适用场景
悲观锁比较适合并发量较小又需要独占读取结果并依赖读取的结果进行判断的业务场景
定义
根据命名即对数据进行操作更新时,对操作持悲观保守的态度,认为产生数据冲突的可能性很大,需要先对请求的数据加锁再进行相关的操作。
实现方式
通过数据库锁机制实现,即对查询语句添加for update关键字。
select * from table where id = 1 for update;
当一个请求A开启事务并执行此sql同时未提交事务时,另一个线程B发起请求,此时B将阻塞在加了锁的查询语句上,直到A请求的事务提交或者回滚,B才会继续执行,保证了访问的隔离性。
悲观锁优缺点分析
优点
每一次行数据的访问都是独占的,只有当正在访问该行数据的请求事务提交以后,其他请求才能依次访问该数据,否则将阻塞等待锁的获取。悲观锁可以严格保证数据访问的安全。
缺点
每次请求都会额外产生加锁的开销且未获取到锁的请求将会阻塞等待锁的获取,在高并发环境下,容易造成大量请求阻塞,影响系统可用性。另外,悲观锁使用不当还可能产生死锁的情况。
我们来看如下情况,以商品表、用户商品列表为例:
系统出现了2个业务操作,
业务A先查询商品表并加锁,根据查询的结果作更新用户商品列表状态字段的操作
select * from product where id = 10 for update;update user_product set status = 2 where user_id = 10001;
业务B先查询用户商品表并加锁,根据查询结果更新商品表剩余数量的操作
select * from user_product where user_id = 10001 for update;
update product set rest = rest - 1 where id = 10
我们看一下产生死锁的过程:A业务操作开启事务,获取product表id=10的行锁,B业务操作接着开启事务并获取user_product表user_id为10001记录的行锁,A继续执行更新操作,需要等待获取user_product表user_id为10001的行锁进入阻塞状态如下图1所示,B继续执行更新操作需要等待获取product表id=10的行锁。此时我们可以发现数据库的状态为A等待的锁被B占住,而B等待的锁被A所占住,双方事务都未提交都在等待对方释放锁,进入一个死循环状态。
数据库(mysql5.7)检测到产生了死锁,自动回滚了B操作的事务,释放了锁。虽然常见数据库如oracle或者mysql都有死锁检测机制,出现死锁数据库会自动回滚一个事务,但是也会造成系统的可用性和稳定性受到影响,我们应该避免在实际应用场景中出现死锁的情况,如上例所示,可以考虑把一个操作改为乐观锁实现或者改变锁的获取顺序使得2个操作都是先获取同一个锁再获取另外一个锁,以此避免死锁的发生。综合数据库悲观锁的特点,在
图左 A操作执行其update操作时等待锁的获取
图右 B操作执行update时,数据库检测到死锁则回滚
浅谈实现数据库乐观锁和悲观锁相关推荐
- mysql共享锁使用方法_浅谈Mysql共享锁、排他锁、悲观锁、乐观锁及其使用场景...
Mysql共享锁.排他锁.悲观锁.乐观锁及其使用场景 一.相关名词 |--表级锁(锁定整个表) |--页级锁(锁定一页) |--行级锁(锁定一行) |--共享锁(S锁,MyISAM 叫做读锁) |-- ...
- mysql锁的应用场景_浅谈Mysql共享锁、排他锁、悲观锁、乐观锁及其使用场景
Mysql共享锁.排他锁.悲观锁.乐观锁及其使用场景 一.相关名词 |--表级锁(锁定整个表) |--页级锁(锁定一页) |--行级锁(锁定一行) |--共享锁(S锁,MyISAM 叫做读锁) |-- ...
- 浅谈面试中常提到的乐观锁与悲观锁
首先来看一下什么是锁? 在并发环境下,会出现多个线程对同一个资源进行争抢的情况,假设A线程对资源正在进行修改,此时B线程此时又对资源进行了修改,这就可能会导致数据不一致的问题.为了解决这个问题,很多编 ...
- 浅谈面试中的乐观锁与悲观锁
乐观锁与悲观锁 一.什么是乐观锁(Optimistic Locking)? 二.什么是悲观锁(Pessimistic Lock)? 悲观锁主要分为共享锁和排他锁 注意 三.并发控制 一.什么是乐观锁( ...
- 浅谈Mysql共享锁、排他锁、悲观锁、乐观锁及其使用场景
Mysql共享锁.排他锁.悲观锁.乐观锁及其使用场景 一.相关名词 |--表级锁(锁定整个表) |--页级锁(锁定一页) |--行级锁(锁定一行) |--共享锁(S锁,MyISAM 叫做读锁) |-- ...
- 浅谈对于乐观锁和悲观锁的理解
今儿回忆了一下关于乐观锁和悲观锁的知识,要唠唠这两个点,那先得知道什么是锁? 1.锁的定义:锁是一种互斥的机制,在多线程环境中实现对资源的协调与控制,凡是有资源被多线程共享,涉及到修改的情况就要考虑锁 ...
- 【转】数据库的乐观锁和悲观锁
[转]数据库的乐观锁和悲观锁 有时候为了得到最大的性能,一般数据库都有并发机制,不过带来的问题就是数据访问的冲突.为了解决这个问题,大多数数据库用的方法就是数据的锁定.所以说,悲观锁和乐观锁主要是用来 ...
- 数据库并发控制,选择乐观锁还是悲观锁?
出处:http://www.cnblogs.com/chenlulouis/ 今天,在这里,我们将讨论的是在实际生产过程中,对于并发控制你是选择乐观锁还是悲观锁.这两种锁各自的应用环境应该怎样选择? ...
- 数据库中的乐观锁与悲观锁详解
目录 悲观锁 乐观锁 悲观锁实现方式 乐观锁实现方式 如何选择 悲观锁 当我们要对一个数据库中的一条数据进行修改的时候,为了避免同时被其他人修改,最好的办法就是直接对该数据进行加锁以防止并发. 这种借 ...
最新文章
- 使用Typescript的巧妙React上下文技巧-不是Redux
- OpenCV 特征点检测
- python怎么查看列表_Python 小白,关于对于列表的+=操作不明白,查了半天也不知道怎么查,并且查询无果,请人指点?...
- 电脑不能打字_电脑拼音打字快速入门秘籍
- 如何给柱状图柱子添加阴影_【LeetCode日记】84. 柱状图中最大的矩形
- Sublime Text 3 初试牛刀
- Retrofit与RXJava整合
- 如何在SQL Server数据库中加密数据
- Ubuntu中Python3找不到_sqlite3模块
- jenkins+phantomjs环境搭建及使用
- Oracle RMAN备份与还原
- 文字图片转换成html代码,IMG4Me 网站在线把文字转换为图片
- 基于springboot电商生鲜购物商城平台设计与实现(含源码+数据库文件)
- 爬楼梯——回溯(python)
- C语言程序出现malloc(): corrupted top size异常中止
- A1、A2、A3、A4纸都是多大尺寸
- python脚本运行越来越慢
- python 三维度显示,Python实现高斯函数的三维显示方法
- 转载:基于AT91RM9200与LINUX2.6.26内核的嵌入式平台开发全过程
- 毕业设计系统需求分析-“大楼要有高低层、单双号电梯,要抗7级地震”-04
热门文章
- 如何正确选择客流统计系统
- 自动驾驶系类文档(二)-- 定位模块(Location)
- 国内主流常用快递物流查询api接口介绍以及demo分享
- 什么是Filter过滤器
- 报错:cannot run program--createprocess error=2,系统找不到指定的文件
- 2022年中国隐私计算综合影响力 TOP10 企业
- 2022年消费金融行业研究报告
- 除和除以以及被除的区别
- 全新C4D必备插件合集他来啦傻瓜式一键安装
- READONLY You can‘t write against a read only replica.