MySQL悲观锁与乐观锁的实现方案


Posted in MySQL onNovember 02, 2021

前言

悲观锁和乐观锁是用来解决并发问题的两种思想,在不同的平台有着各自的实现。例如在Java中,synchronized就可以认为是悲观锁的实现(不严谨,有锁升级的过程,升级到重量级锁才算),Atomic***原子类可以认为是乐观锁的实现。

悲观锁

具有强烈的独占和排他特性,在整个处理过程中将数据处于锁定状态,一般是通过系统的互斥量来实现。当其他线程想要获取锁时会被阻塞,直到持有锁的线程释放锁。

乐观锁

对数据的修改和访问持乐观态度,假设不会发生冲突,只有当数据提交更新时才会对数据冲突与否进行检测,如果没有冲突则顺利提交更新,否则快速失败,返回一个错误给用户,让用户选择接下来该如何去做,一般来说失败后会继续重试,直到提交更新成功为止。

MySQL本身就支持锁机制,例如我们有一个「先查再写」的需求,我们希望整个流程是一个原子操作,中间不能被打断,这时候就可以通过给查询的数据行加「排他锁」来实现。只要当前事务不释放锁,其他事务要想获得排他锁,MySQL就会将其阻塞,直到当前事务释放锁。这种MySQL底层的排他锁就称作「悲观锁」。

MySQL本身不提供乐观锁的功能,需要开发者自己实现。普遍的做法是在表中加一个version列,用来标记数据行的版本,当我们需要更新数据时,必须比对version版本,version一致说明这个期间数据没有被其他事务修改过,否则说明数据已经被其他事务修改,需要自旋重试了。

实战

假设数据库有两张表:商品表和订单表。

用户下单后需要执行两个操作:

  1. 商品表减去库存。
  2. 订单表创建一条记录。

初始数据:ID为1的商品有100的库存,订单表数据为空。

客户端启动10个线程并发下单,分别在无锁、悲观锁、乐观锁的场景下有哪些表现。

如下是创建表的sql语句:

-- 商品表
CREATE TABLE `goods` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT,
  `goods_name` varchar(50) NOT NULL,
  `price` decimal(10,2) NOT NULL,
  `stock` int(11) DEFAULT '0',
  `version` int(10) unsigned NOT NULL DEFAULT '0',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8

-- 订单表
CREATE TABLE `t_order` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT,
  `goods_id` bigint(20) NOT NULL,
  `order_time` datetime NOT NULL,
  PRIMARY KEY (`id`) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8

1、无锁

不做任何处理。

// 下单
private boolean order(){
    Goods goods = goodsMapper.selectById(1L);
    boolean success = false;
    if (goods.getStock() > 0) {
        goods.setStock(goods.getStock() - 1);
        // 更新库存
        goodsMapper.updateById(goods);
        // 创建订单
        orderMapper.save(goods.getId());
        success = true;
    }
    return success;
}

控制台输出结果:

MySQL悲观锁与乐观锁的实现方案

2、悲观锁

查询商品时加FOR UPDATE,给数据行加排他锁,这样其他线程再查询时就会被阻塞,直到当前线程的事务提交并释放锁,其他线程才能继续下单。这种方式并发性能不高。

sql语句

@Select("SELECT * FROM goods WHERE id = #{id} FOR UPDATE")
Goods selectForUpdate(Long id);

控制台输出结果:

MySQL悲观锁与乐观锁的实现方案

注意:FOR UPDATE必须在事务中才有效,查询和更新必须在同一个事务中!!!

3、乐观锁

实现思路是:每次更新时校验版本号,如果版本号一致说明期间数据没有被其他线程改过,当前线程可以正常提交更新,否则说明数据已经被其他线程改过了,当前线程需要自旋重试,直到业务成功为止。

更新数据的同时版本号必须自增!!!

@Update("UPDATE goods SET stock = #{stock},version = version+1 WHERE id = #{id} AND version = #{version}")
int updateByVersion(Long id, Integer stock, Integer version);

业务代码

boolean order(){
    Goods goods = goodsMapper.selectById(1L);
    boolean success = false;
    if (goods.getStock() > 0) {
        goods.setStock(goods.getStock() - 1);
        // 更新库存,带上版本号
        int result = goodsMapper.updateByVersion(goods.getId(), goods.getStock(), goods.getVersion());
        if (result <= 0) {
            // 更新失败,说明期间数据已经被其他线程修改,需要递归重试
            return order();
        }
        // 创建订单
        orderMapper.save(goods.getId());
        success = true;
    }
    return success;
}

控制台输出结果:

MySQL悲观锁与乐观锁的实现方案

总结

到此这篇关于MySQL悲观锁与乐观锁方案的文章就介绍到这了,更多相关MySQL悲观锁与乐观锁内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

MySQL 相关文章推荐
浅谈mysql执行过程以及顺序
May 12 MySQL
.Net Core导入千万级数据至Mysql的步骤
May 24 MySQL
详细谈谈MYSQL中的COLLATE是什么
Jun 11 MySQL
MySQL 那些常见的错误设计规范,你都知道吗
Jul 16 MySQL
MySQL8.0的WITH查询详情
Aug 30 MySQL
基于MySql验证的vsftpd虚拟用户
Nov 07 MySQL
MySQL中CURRENT_TIMESTAMP的使用方式
Nov 27 MySQL
解决MySQL添加新用户-ERROR 1045 (28000)的问题
Mar 03 MySQL
Linux系统下MySQL配置主从分离的步骤
Mar 21 MySQL
你真的会用Mysql的explain吗
Mar 31 MySQL
Mysql InnoDB 的内存逻辑架构
May 06 MySQL
MySQL表字段数量限制及行大小限制详情
Jul 23 MySQL
SpringBoot连接MySQL获取数据写后端接口的操作方法
Mysql排序的特性详情
Nov 01 #MySQL
MySQL分区表实现按月份归类
Nov 01 #MySQL
MySQL数据库10秒内插入百万条数据的实现
MySQL面试题讲解之如何设置Hash索引
MySQL对数据表已有表进行分区表的实现
Nov 01 #MySQL
mysql分表之后如何平滑上线详解
Nov 01 #MySQL
You might like
浅谈电磁辐射对健康的影响
2021/03/01 无线电
php中导出数据到excel时数字变为科学计数的解决方法
2013/02/03 PHP
PHP PDO和消息队列的个人理解与应用实例分析
2019/11/25 PHP
jQuery中filter()和find()的区别深入了解
2013/09/25 Javascript
JS实现的一个简单的Autocomplete自动完成例子
2014/04/16 Javascript
Javascript 绘制 sin 曲线过程附图
2014/08/21 Javascript
浅谈jquery事件处理
2015/04/24 Javascript
JavaScript中获取Radio被选中的值
2015/11/11 Javascript
js实现九宫格拼图小游戏
2017/02/13 Javascript
旺旺在线客服代码 旺旺客服代码生成器
2018/01/09 Javascript
vue实现的上传图片到数据库并显示到页面功能示例
2018/03/17 Javascript
JS实现深度优先搜索求解两点间最短路径
2019/01/17 Javascript
基于Angular 8和Bootstrap 4实现动态主题切换的示例代码
2020/02/11 Javascript
vue实现简单加法计算器
2020/10/22 Javascript
[02:10]DOTA2 TI10勇士令状玩法及不朽Ⅰ展示:焕新世界,如你所期
2020/05/29 DOTA
Python3简单实例计算同花的概率代码
2017/12/06 Python
Python测试网络连通性示例【基于ping】
2018/08/03 Python
Python3爬虫使用Fidder实现APP爬取示例
2018/11/27 Python
Python实现去除列表中重复元素的方法总结【7种方法】
2019/02/16 Python
Python脚本修改阿里云的访问控制列表的方法
2019/03/08 Python
使用Django搭建web服务器的例子(最最正确的方式)
2019/08/29 Python
python 发送json数据操作实例分析
2019/10/15 Python
Python文件操作函数用法实例详解
2019/12/24 Python
TensorFlow自定义损失函数来预测商品销售量
2020/02/05 Python
一篇文章搞懂python的转义字符及用法
2020/09/03 Python
详解Django ORM引发的数据库N+1性能问题
2020/10/12 Python
Python Sqlalchemy如何实现select for update
2020/10/12 Python
浅谈matplotlib默认字体设置探索
2021/02/03 Python
美国汽配连锁巨头Pep Boys官网:轮胎更换、汽车维修服务和汽车零部件
2017/01/14 全球购物
经典优秀个人求职自荐信格式
2013/09/25 职场文书
师范应届生求职信
2013/11/15 职场文书
优秀学生干部个人事迹材料
2014/06/02 职场文书
试用期员工工作自我评价
2014/09/10 职场文书
2014年手术室工作总结
2014/11/26 职场文书
Python中json.load()和json.loads()有哪些区别
2021/06/07 Python
MySQL定时备份数据库(全库备份)的实现
2021/09/25 MySQL