Mysql 系列文章主页

===============

刚开始学习 Mysql 锁的时候,觉得 Mysql 使用的是行锁,再加上其默认的可重复读的隔离级别,那就应该能够自动解决并发事务更新的问题。可事实上,并不能解决!

可是,为什么不能解决呢?

带着问题,自己写了个简单Demo,来测试并分析其中道理:

Demo 代码路径:https://github.com/cyhbyw/cyh_Spring_IsolationConcurrencyTransaction

Demo 工程名称:usingMybatis

0 准备工作

工程中所有文件如下图所示;

代码运行前,注意设置数据库的用户名和密码,同时,需要运行 test.sql 文件中的脚本用以初始化数据;

1 测试

1.1 业务代码

     @Transactional(rollbackFor = RuntimeException.class)
public void increaseMoney(Integer id) {
Employee employee = employeeMapper.findById(id);
final Integer oldMoney = employee.getMoney();
LOGGER.info("oldMoney: {}", oldMoney);
employee.setMoney(oldMoney + 1);
employeeMapper.updateEmployee(employee);
}

代码说明:

  • 第1行,添加事务
  • 第3行,取出数据
  • 第4,5行,取出旧的Money值并记录日志
  • 第6行,为 Employee 对象设置新的值(就是+1操作)
  • 第7行,执行Sql更新

1.2 测试代码

     private void increaseMoney() {
int threadCount = 100;
while (threadCount-- > 0) {
new Thread(() -> employeeService.increaseMoney(1)).start();
}
}

代码说明:

  • 启了 100 根线程去调用业务方法
  • 每个线程将 Money+1,理论上,最后的Money=100(初始值为0)

1.3 测试结果

预期结果:100,实际结果:22

提示:22这个值不固定(因为是多线程访问嘛,而线程的执行是由CPU决定的,具有一定波动性)

1.4 程序日志

 2018-04-25 11:32:05,416  INFO [Thread-83] (EmployeeService.java:27) - oldMoney: 0
2018-04-25 11:32:05,416 INFO [Thread-18] (EmployeeService.java:27) - oldMoney: 0
2018-04-25 11:32:05,416 INFO [Thread-19] (EmployeeService.java:27) - oldMoney: 0
2018-04-25 11:32:05,416 INFO [Thread-54] (EmployeeService.java:27) - oldMoney: 0
2018-04-25 11:32:05,416 INFO [Thread-15] (EmployeeService.java:27) - oldMoney: 0
2018-04-25 11:32:05,416 INFO [Thread-52] (EmployeeService.java:27) - oldMoney: 0
2018-04-25 11:32:05,416 INFO [Thread-17] (EmployeeService.java:27) - oldMoney: 0
2018-04-25 11:32:05,416 INFO [Thread-55] (EmployeeService.java:27) - oldMoney: 0
2018-04-25 11:32:05,416 INFO [Thread-53] (EmployeeService.java:27) - oldMoney: 0
2018-04-25 11:32:05,416 INFO [Thread-20] (EmployeeService.java:27) - oldMoney: 0
2018-04-25 11:32:05,416 INFO [Thread-43] (EmployeeService.java:27) - oldMoney: 0
2018-04-25 11:32:05,416 INFO [Thread-2] (EmployeeService.java:27) - oldMoney: 0
2018-04-25 11:32:05,416 INFO [Thread-56] (EmployeeService.java:27) - oldMoney: 0
2018-04-25 11:32:05,416 INFO [Thread-8] (EmployeeService.java:27) - oldMoney: 0
2018-04-25 11:32:05,416 INFO [Thread-16] (EmployeeService.java:27) - oldMoney: 0
2018-04-25 11:32:05,462 INFO [Thread-14] (EmployeeService.java:27) - oldMoney: 1
2018-04-25 11:32:05,527 INFO [Thread-27] (EmployeeService.java:27) - oldMoney: 2
2018-04-25 11:32:05,534 INFO [Thread-26] (EmployeeService.java:27) - oldMoney: 2
2018-04-25 11:32:05,534 INFO [Thread-22] (EmployeeService.java:27) - oldMoney: 2
2018-04-25 11:32:05,535 INFO [Thread-23] (EmployeeService.java:27) - oldMoney: 2
2018-04-25 11:32:05,534 INFO [Thread-24] (EmployeeService.java:27) - oldMoney: 2
2018-04-25 11:32:05,538 INFO [Thread-29] (EmployeeService.java:27) - oldMoney: 3
2018-04-25 11:32:05,538 INFO [Thread-25] (EmployeeService.java:27) - oldMoney: 3
2018-04-25 11:32:05,540 INFO [Thread-32] (EmployeeService.java:27) - oldMoney: 3
2018-04-25 11:32:05,541 INFO [Thread-30] (EmployeeService.java:27) - oldMoney: 3
2018-04-25 11:32:05,540 INFO [Thread-28] (EmployeeService.java:27) - oldMoney: 3
2018-04-25 11:32:05,542 INFO [Thread-31] (EmployeeService.java:27) - oldMoney: 3
2018-04-25 11:32:05,542 INFO [Thread-33] (EmployeeService.java:27) - oldMoney: 3
2018-04-25 11:32:05,548 INFO [Thread-37] (EmployeeService.java:27) - oldMoney: 4
2018-04-25 11:32:05,550 INFO [Thread-21] (EmployeeService.java:27) - oldMoney: 4
2018-04-25 11:32:05,550 INFO [Thread-34] (EmployeeService.java:27) - oldMoney: 4
2018-04-25 11:32:05,575 INFO [Thread-39] (EmployeeService.java:27) - oldMoney: 5
2018-04-25 11:32:05,575 INFO [Thread-38] (EmployeeService.java:27) - oldMoney: 5
2018-04-25 11:32:05,576 INFO [Thread-41] (EmployeeService.java:27) - oldMoney: 5
2018-04-25 11:32:05,575 INFO [Thread-40] (EmployeeService.java:27) - oldMoney: 5
2018-04-25 11:32:05,577 INFO [Thread-42] (EmployeeService.java:27) - oldMoney: 5
2018-04-25 11:32:05,577 INFO [Thread-44] (EmployeeService.java:27) - oldMoney: 6
2018-04-25 11:32:05,578 INFO [Thread-45] (EmployeeService.java:27) - oldMoney: 6
2018-04-25 11:32:05,602 INFO [Thread-35] (EmployeeService.java:27) - oldMoney: 7
2018-04-25 11:32:05,603 INFO [Thread-13] (EmployeeService.java:27) - oldMoney: 7
2018-04-25 11:32:05,604 INFO [Thread-36] (EmployeeService.java:27) - oldMoney: 7
2018-04-25 11:32:05,604 INFO [Thread-12] (EmployeeService.java:27) - oldMoney: 8
2018-04-25 11:32:05,605 INFO [Thread-10] (EmployeeService.java:27) - oldMoney: 8
2018-04-25 11:32:05,606 INFO [Thread-9] (EmployeeService.java:27) - oldMoney: 9
2018-04-25 11:32:05,606 INFO [Thread-6] (EmployeeService.java:27) - oldMoney: 9
2018-04-25 11:32:05,606 INFO [Thread-1] (EmployeeService.java:27) - oldMoney: 9
2018-04-25 11:32:05,644 INFO [Thread-47] (EmployeeService.java:27) - oldMoney: 10
2018-04-25 11:32:05,644 INFO [Thread-48] (EmployeeService.java:27) - oldMoney: 10
2018-04-25 11:32:05,644 INFO [Thread-46] (EmployeeService.java:27) - oldMoney: 10
2018-04-25 11:32:05,649 INFO [Thread-49] (EmployeeService.java:27) - oldMoney: 10
2018-04-25 11:32:05,650 INFO [Thread-75] (EmployeeService.java:27) - oldMoney: 10
2018-04-25 11:32:05,651 INFO [Thread-76] (EmployeeService.java:27) - oldMoney: 10
2018-04-25 11:32:05,651 INFO [Thread-50] (EmployeeService.java:27) - oldMoney: 10
2018-04-25 11:32:05,686 INFO [Thread-79] (EmployeeService.java:27) - oldMoney: 11
2018-04-25 11:32:05,686 INFO [Thread-78] (EmployeeService.java:27) - oldMoney: 11
2018-04-25 11:32:05,686 INFO [Thread-77] (EmployeeService.java:27) - oldMoney: 11
2018-04-25 11:32:05,693 INFO [Thread-80] (EmployeeService.java:27) - oldMoney: 11
2018-04-25 11:32:05,693 INFO [Thread-100] (EmployeeService.java:27) - oldMoney: 11
2018-04-25 11:32:05,693 INFO [Thread-81] (EmployeeService.java:27) - oldMoney: 11
2018-04-25 11:32:05,694 INFO [Thread-51] (EmployeeService.java:27) - oldMoney: 11
2018-04-25 11:32:05,694 INFO [Thread-82] (EmployeeService.java:27) - oldMoney: 11
2018-04-25 11:32:05,737 INFO [Thread-59] (EmployeeService.java:27) - oldMoney: 12
2018-04-25 11:32:05,738 INFO [Thread-58] (EmployeeService.java:27) - oldMoney: 12
2018-04-25 11:32:05,739 INFO [Thread-57] (EmployeeService.java:27) - oldMoney: 12
2018-04-25 11:32:05,746 INFO [Thread-60] (EmployeeService.java:27) - oldMoney: 12
2018-04-25 11:32:05,746 INFO [Thread-99] (EmployeeService.java:27) - oldMoney: 12
2018-04-25 11:32:05,746 INFO [Thread-62] (EmployeeService.java:27) - oldMoney: 12
2018-04-25 11:32:05,748 INFO [Thread-61] (EmployeeService.java:27) - oldMoney: 12
2018-04-25 11:32:05,786 INFO [Thread-63] (EmployeeService.java:27) - oldMoney: 13
2018-04-25 11:32:05,786 INFO [Thread-66] (EmployeeService.java:27) - oldMoney: 13
2018-04-25 11:32:05,786 INFO [Thread-64] (EmployeeService.java:27) - oldMoney: 13
2018-04-25 11:32:05,792 INFO [Thread-65] (EmployeeService.java:27) - oldMoney: 13
2018-04-25 11:32:05,792 INFO [Thread-97] (EmployeeService.java:27) - oldMoney: 13
2018-04-25 11:32:05,792 INFO [Thread-98] (EmployeeService.java:27) - oldMoney: 13
2018-04-25 11:32:05,792 INFO [Thread-68] (EmployeeService.java:27) - oldMoney: 13
2018-04-25 11:32:05,792 INFO [Thread-96] (EmployeeService.java:27) - oldMoney: 13
2018-04-25 11:32:05,839 INFO [Thread-95] (EmployeeService.java:27) - oldMoney: 14
2018-04-25 11:32:05,840 INFO [Thread-92] (EmployeeService.java:27) - oldMoney: 14
2018-04-25 11:32:05,839 INFO [Thread-93] (EmployeeService.java:27) - oldMoney: 14
2018-04-25 11:32:05,860 INFO [Thread-94] (EmployeeService.java:27) - oldMoney: 14
2018-04-25 11:32:05,862 INFO [Thread-67] (EmployeeService.java:27) - oldMoney: 14
2018-04-25 11:32:05,861 INFO [Thread-89] (EmployeeService.java:27) - oldMoney: 14
2018-04-25 11:32:05,867 INFO [Thread-91] (EmployeeService.java:27) - oldMoney: 14
2018-04-25 11:32:05,887 INFO [Thread-90] (EmployeeService.java:27) - oldMoney: 15
2018-04-25 11:32:05,890 INFO [Thread-88] (EmployeeService.java:27) - oldMoney: 16
2018-04-25 11:32:05,893 INFO [Thread-85] (EmployeeService.java:27) - oldMoney: 17
2018-04-25 11:32:05,893 INFO [Thread-86] (EmployeeService.java:27) - oldMoney: 17
2018-04-25 11:32:05,894 INFO [Thread-84] (EmployeeService.java:27) - oldMoney: 17
2018-04-25 11:32:05,894 INFO [Thread-87] (EmployeeService.java:27) - oldMoney: 18
2018-04-25 11:32:05,895 INFO [Thread-5] (EmployeeService.java:27) - oldMoney: 18
2018-04-25 11:32:05,896 INFO [Thread-4] (EmployeeService.java:27) - oldMoney: 19
2018-04-25 11:32:05,935 INFO [Thread-71] (EmployeeService.java:27) - oldMoney: 20
2018-04-25 11:32:05,935 INFO [Thread-70] (EmployeeService.java:27) - oldMoney: 20
2018-04-25 11:32:05,936 INFO [Thread-73] (EmployeeService.java:27) - oldMoney: 20
2018-04-25 11:32:05,941 INFO [Thread-69] (EmployeeService.java:27) - oldMoney: 20
2018-04-25 11:32:05,941 INFO [Thread-74] (EmployeeService.java:27) - oldMoney: 20
2018-04-25 11:32:05,941 INFO [Thread-72] (EmployeeService.java:27) - oldMoney: 20
2018-04-25 11:32:05,941 INFO [Thread-7] (EmployeeService.java:27) - oldMoney: 20
2018-04-25 11:32:05,978 INFO [Thread-11] (EmployeeService.java:27) - oldMoney: 21
2018-04-25 11:32:05,978 INFO [Thread-3] (EmployeeService.java:27) - oldMoney: 21

日志分析:

  1. 日志中第1到15行,它们的 oldMoney 全部等于0
  2. 说明对应的15根线程在同一时间读到了相同的值0
  3. 这15根线程给Money设置的新值都是 0+1=1
  4. 这15根线程再分别执行Sql更新操作,期间可能会竞争Mysql的行锁

2 不能解决的原因分析

现在来分析一下为什么Mysql的行锁加上可重复读不能解决并发事务的更新问题

可重复读:表示的是在同一个事务内,所有的读取操作返回相同的数据,也就是不感知外部事务对数据的变更。假设事务A是当前事务,事务B是外部事务,事务A现在读取到 Money=10,接下来,事务B也读到值为10同时将其更新为11,但是对于事务A来说,它不能感知到外部事务对数据的变更,所以,事务A还是错误地认为当前 Money=10 并将其更新为11,于是,事务A覆盖了事务B的提交,从而造成更新丢失。

至于行锁,这只是用来保证并发事务更新时的先后顺序。比如,事务B在将Money=10更新为11时,如果正在这个更新期间(假设更新时间比较长)事务A也想做更新操作,则事务A将被阻塞,直到事务B更新完成后事务A才能更新,它与数据本身(事务A的、事务B的)没有任何关系。

由此可见,虽然 Mysql 是行锁且默认是可重复读,但是对于并发事务的更新操作,仍然会出现更新丢失问题,解决办法是——锁。

Mysql锁机制--并发事务带来的更新丢失问题的更多相关文章

  1. mysql锁机制和事务隔离

    mysql事务 1.InnoDB事务原理 事务(Transaction)是数据库区别于文件系统的重要特性之一,事务会把数据库从一种一致性状态转换为另一种一致性状态. 在数据库提交时,可以确保要么所有修 ...

  2. Mysql 锁机制和事务

    InnoDB 锁机制 InnoDB存储引擎支持行级锁 其大类可以细分为共享锁和排它锁两类 共享锁(S):允许拥有共享锁的事务读取该行数据.当一个事务拥有一行的共享锁时,另外的事务可以在同一行数据也获得 ...

  3. Mysql锁机制和事务控制

    如何加锁 锁定表的语法:    LOCK TABLES    tbl_name [AS alias] {READ [LOCAL] | [LOW_PRIORITY] WRITE}    [, tbl_n ...

  4. Mysql锁机制--乐观锁 & 悲观锁

    Mysql 系列文章主页 =============== 从 这篇 文章中,我们知道 Mysql 并发事务会引起更新丢失问题,解决办法是锁.所以本文将对锁(乐观锁.悲观锁)进行分析. 第一部分 悲观锁 ...

  5. (三)MySQL锁机制 + 事务

    转: (三)MySQL锁机制 + 事务 表锁(偏读) 偏向MyISAM存储引擎.开销小,加锁快,无死锁,锁定粒度大,发生锁冲突的概率最高,并发最低. 查看当前数据库中表的上锁情况,0表示未上锁. sh ...

  6. mysql锁机制(转载)

    锁是计算机协调多个进程或线程并发访问某一资源的机制 .在数据库中,除传统的 计算资源(如CPU.RAM.I/O等)的争用以外,数据也是一种供许多用户共享的资源.如何保证数据并发访问的一致性.有效性是所 ...

  7. 再谈mysql锁机制及原理—锁的诠释

    加锁是实现数据库并发控制的一个非常重要的技术.当事务在对某个数据对象进行操作前,先向系统发出请求,对其加锁.加锁后事务就对该数据对象有了一定的控制,在该事务释放锁之前,其他的事务不能对此数据对象进行更 ...

  8. mysql锁机制 读书笔记

    目录 MySQL锁机制 1.什么是锁 2.lock与latch 3.InnoDB存储引擎中的锁 3.1锁的类型 3.2 一致性非锁定读 3.3 一致性锁定读 4 锁的算法 4.1行锁的3中算法 4.2 ...

  9. Mysql锁机制介绍

    Mysql锁机制介绍 一.概况MySQL的锁机制比较简单,其最显著的特点是不同的存储引擎支持不同的锁机制.比如,MyISAM和MEMORY存储引擎采用的是表级锁(table-level locking ...

随机推荐

  1. Javascript 装饰器极速指南

    pablo.png Decorators 是ES7中添加的JavaScript新特性.熟悉Typescript的同学应该更早的接触到这个特性,TypeScript早些时候已经支持Decorators的 ...

  2. C# 启动 SQL Server 服务

    //首先要添加 System.ServiceProcess.dll 引用 ServiceController sc = new ServiceController("MSSQLSERVER& ...

  3. 增加Linux虚拟机的硬盘空间

    原配置为40G,现需要增加到60G,操作方法如下: 一.虚拟机关机,在编辑设置里调整硬盘空间到60G 二.虚拟机开机,扩展硬盘空间 1.安装gparted,命令如下 sudo apt-get inst ...

  4. Python-函数-Day4

    1.函数 1.1.集合 主要作用: 去重 关系测试, 交集\差集\并集\反向(对称)差集 a = {1,2,3,4} b ={3,4,5,6} a {1, 2, 3, 4} type(a) <c ...

  5. axure 预览"HTTP/1.1 302 Found"

    使用Axure编辑原型时,点击预览出现"HTTP/1.1 302 Found" 第一想到的就是重新安装Axure和检查原型文件是否损坏,验证后证明前Axure和.rp文件都是完好的 ...

  6. HTTP与私有二进制协议之间的区别

    简单的文本协议.二进制协议 写网络程序躲不过协议,协议其实就是定义了消息的格式,以及消息是如何交换的.协议可简单可复杂,复杂精密如TCP协议,简单奔放如HTTP的协议.这里将我所接触到的协议稍微总结一 ...

  7. Docker学习笔记 - Docker容器与外部网络的连接

    学习目的: ip_forward 包过滤防护墙 iptables 允许端口映射访问 限制ip访问容器 1.ip_forward 控制系统是否会转发流量 检查linux系统转发是否开启命令:sysctl ...

  8. Mybatis自动生成Xml文件,针对字段类型为text等会默认产生XXXXWithBlobs的方法问题

    默认情况下产生的Mapper.xml里面存在: 需要修改generatorConfiguration.xml,里面的table加属性,如: <table domainObjectName=&qu ...

  9. Mysql 库表

    create database student_info default character set utf8 collate utf8_general_ci; ------------------- ...

  10. NetSNMP开源代码学习——mib扩展

    扩展MIB库关于MIB库的扩展网络文章非常多,这里我主要参考了http://blog.csdn.net/qq_27204267/article/details/51595708,这篇文章介绍的比较简单 ...