mysql的事务隔离级别

事先声明:都是个人理解,若有偏差,勿怪

1.事务是个什么东西?
一系列操作的集合;一条事务包含多个数据库操作(n条增删该查的statement语句)
2.事务提交是个什么鬼?
事务提交前,操作结果都是存储在临时库中,此时的数据改变对我们来说依旧可见,若事务出现错误或连接超时,则会回滚
事务提交后,操作结果才会存储到数据库中,不会回滚

3.如果使用事务?

3.1 事务也需要管理器(管理器是个什么鬼?不知道,反正是用来管理事务的),spring中事务管理器的顶层接口是PlatformTransactionManager。而mybatis框架用的是DataSourceTransactionManager(最常用),它是对PlatformTransactionManager接口的实现。(这就是个知识点,没啥大用处,如果面试官太jian,可能会问道)

3.2 事务的隔离级别:

读取未提交:A事务 能读取 B事务 未提交的数据/临时库中的数据(如果两个事务在执行过程中都没啥问题,自然皆大欢喜,此隔离级别就是最牛逼的隔离级别。但是如果B事务报错回滚,而A事务读取了B事务回滚前的数据(该数据由于B事务的回滚,已成为垃圾数据),就会造成A事务出现**脏读现象**,导致数据逻辑上的出错,即最后结果与真实结果不一样)

读取已提交:A事务只能读取B事务已经提交的数据/非临时库中的数据(如果两个事务在执行过程中都没啥问题,**可能最后的结果依旧有问题**。场景:数据库中 商品库存为1。B事务扣减库存但未提交 临时库中的库存变为0,此时A事务不能读取临时库库存只能读取数据库库存为1。接着B事务提交数据,数据库库存变为0,A事务扣减库存发现竟然无法扣减(不可/无法重复读现象)。这是因为A事务只读取了一次库存(无法重复读取),如果能够在扣减前再读取一次事务(显然是无法实现的,就算A事务读取操作和扣减操作中间没有任何其他操作,但是由于并行执行,B事务的操作可能就会在A事务的这两操作之间执行,所以说如果采用了该隔离级别,不可重复读的现象是无法消除的)就能够完美解决这一现象。

可重复读:这是为了克服不可重复读现象而产生的隔离级别。B事务在读取库存后,A事务就不能读取库存,直到B事务提交,A事务才能读取。也就是说,如果某一个事务读取了某个资源,该资源就不能被其他事务读取,直到该事务提交(看着也不像重复读取啊!为啥取名叫可重复读呢?百思不得解。觉着更像是资源占用,不释放)。但是该隔离级别有可能造成**幻读**(场景:库存10,交易记录5。B事务读取库存10,A事务读取交易记录统计记录数5。B事务扣减库存,插入一笔交易记录提交事务。A打印交易记录共6条。A统计的记录数和打印出来的记录数不一致)

串行化:数据库最高隔离级别(性能最差),要求所有的sql语句按顺序执行(不会并发执行)

总结一下造成各个现象的根本原因:
脏读:事务回滚造成垃圾数据
不可重复读:由于并发机制,事务的读取操作和非读取操作之间执行了其他事务的非读取操作
幻读:和不可重复读相似,事务读取操作和非读取操作执行了其他事务的非读取操作,但是幻读更侧重于记录数统计上的数据错误,而不可重复读更侧重于数据值的错误。

3.3 使用隔离级别(mysql默认隔离级别为可重复读,oracle默认为读取已提交)

@Transactional(isolation = Isolation.READ_UNCOMMITTED)
在这里插入图片描述

4.@Transactional注解在某一场景下会失效
类自身方法之间的调用,@Transactional注解会失效。原因:数据库事务的原理帮助其实就是aop,而aop的原理就是动态代理,而在自调用的过程中,是类自身的调用,而不是代理对象去调用。

可能表达的不清楚,因为我也有点生疏,只是把我了解的以及理解的部分写出来,希望对大家有帮助

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值