两个事务先后进行数据更新 第二次事务查询的数据是错误的

问题描述

A方法包含B方法,A方法先更新数据,再执行B方法查询数据。AB方法均有事务管理。

事务A 事务B 对 某一账户进行更新

具体逻辑就是 
先执行事务 A
开启事务
获取账户余额 查询得到 x
更新账户余额为 x + 1
事务提交

再执行事务B

事务B 里面 也是按照这个逻辑来 但是 获取账户余额查询得到的是 x 而不是 x + 1

解决方法

检查事务隔离级别的设置。

SELECT @@GLOBAL.tx_isolation, @@tx_isolation;
 

在REPEATABLE-READ情况下

如果第二个事务如果在第一个事务commit前开启,那读到的是原先的数据

在Read-Committed的情况下

当第一个事务commit后,第二个事务会读到提交后的数据



可以看看第二个事务的开启是在第一个事务提交前还是提交后

代码如何解决:

新建C方法,将更新方法A方法,查询方法B方法分别放进C方法,这样AB方法就是单独的事务了。

附录:

1.查看当前会话隔离级别

select @@tx_isolation;

2.查看系统当前隔离级别

select @@global.tx_isolation;

3.设置当前会话隔离级别

set session transaction isolatin level repeatable read;

4.设置系统当前隔离级别

set global transaction isolation level repeatable read;

5.命令行,开始事务时

set autocommit=off 或者 start transaction

关于隔离级别的理解

1.read uncommitted

可以看到未提交的数据(脏读),举个例子:别人说的话你都相信了,但是可能他只是说说,并不实际做。

2.read committed

读取提交的数据。但是,可能多次读取的数据结果不一致(不可重复读,幻读)。用读写的观点就是:读取的行数据,可以写。

3.repeatable read(MySQL默认隔离级别)

可以重复读取,但有幻读。读写观点:读取的数据行不可写,但是可以往表中新增数据。在MySQL中,其他事务新增的数据,看不到,不会产生幻读。采用多版本并发控制(MVCC)机制解决幻读问题。

4.serializable

可读,不可写。像java中的锁,写数据必须等待另一个事务结束。

发布了203 篇原创文章 · 获赞 136 · 访问量 42万+

猜你喜欢

转载自blog.csdn.net/u012045045/article/details/101054501