Spring事务的传播机制

每天早上七点三十,准时推送干货

我们都知道,在数据库中有这事务的存在,但是更多的时候,我们还会被经常问到这个关于 Spring 中 事务的各种行为,又或者说是传播机制,或者隔离级别等内容,那么面试的时候我们应该怎么去回答这个 Spring 事务的传播机制以及隔离级别呢?

什么是 Spring 的事务

事务是逻辑处理原⼦性的保证⼿段,通过使⽤事务控制,可以极⼤的避免出现逻辑处理失败导致的脏数 据等问题。

事务最重要的两个特性,是事务的传播级别和数据隔离级别。传播级别定义的是事务的控制范围,事务 隔离级别定义的是事务在数据库读写⽅⾯的控制范围。

Spring事务传播机制

PROPAGATION_REQUIRED

Spring默认的事务传播级别,使⽤该级别的特点是,如果上下⽂中 已经存在事务,那么就加⼊到事务中执⾏,如果当前上下⽂中不存在事务,则新建事务执⾏。所以这个 级别通常能满⾜处理⼤多数的业务场景。

PROPAGATION_SUPPORTS

从字⾯意思就知道,supports,⽀持,该传播级别的特点是,如果上下⽂存在事务,则⽀持事务加⼊事务,如果没有事务,则使⽤⾮事务的⽅式执⾏。所以说,并⾮所有的包在transactionTemplate.execute中的代码都会有事务⽀持。这个通常是⽤来处理那些并⾮原⼦性的⾮核⼼业务逻辑操作。应⽤场景较少。

PROPAGATION_MANDATORY

该级别的事务要求上下⽂中必须要存在事务,否则就会抛出异常!配置该⽅式的传播级别是有效的控制上下⽂调⽤代码遗漏添加事务控制的保证⼿段。⽐如⼀段代码不能单独被调⽤执⾏,但是⼀旦被调⽤,就必须有事务包含的情况,就可以使⽤这个传播级别。

PROPAGATION_REQUIRES_NEW

从字⾯即可知道,new,每次都要⼀个新事务,该传播级别的特点是,每次都会新建⼀个事务,并且同时将上下⽂中的事务挂起,执⾏当前新建事务完成以后,上下⽂事务恢复再执⾏。

这是⼀个很有⽤的传播级别,举⼀个应⽤场景:现在有⼀个发送100个红包的操作,在发送之前,要做 ⼀些系统的初始化、验证、数据记录操作,然后发送100封红包,然后再记录发送⽇志,发送⽇志要求 100%的准确,如果⽇志不准确,那么整个⽗事务逻辑需要回滚。

怎么处理整个业务需求呢?就是通过这个PROPAGATION_REQUIRES_NEW 级别的事务传播控制就可以 完成。发送红包的⼦事务不会直接影响到⽗事务的提交和回滚。

PROPAGATION_NOT_SUPPORTED

这个也可以从字⾯得知,not supported ,不⽀持,当前级别的特点就是上下⽂中存在事务,则挂起事务,执⾏当前逻辑,结束后恢复上下⽂的事务。

这个级别有什么好处?

可以帮助你将事务极可能的缩⼩。我们知道⼀个事务越⼤,它存在的⻛险也就越多。所以在处理事务的过程中,要保证尽可能的缩⼩范围。⽐如⼀段代码,是每次逻辑操作都必须调⽤的,⽐如循环1000次的某个⾮核⼼业务逻辑操作。这样的代码如果包在事务中,势必造成事务太⼤,导致出现⼀些难以考虑周全的异常情况。所以这个事务这个级别的传播级别就派上⽤场了。

PROPAGATION_NEVER

该事务更严格,上⾯⼀个事务传播级别只是不⽀持⽽已,有事务就挂 起,⽽PROPAGATION_NEVER传播级别要求上下⽂中不能存在事务,⼀旦有事务,就抛出runtime异 常,强制停⽌执⾏!这个级别上辈⼦跟事务有仇。

PROPAGATION_NESTED

字⾯也可知道,nested,嵌套级别事务。该传播级别特征是,如果上 下⽂中存在事务,则嵌套事务执⾏,如果不存在事务,则新建事务

Spring 事务隔离级别

Serializable

最严格的级别,事务串⾏执⾏,资源消耗最⼤;

REPEATABLE READ

保证了⼀个事务不会修改已经由另⼀个事务读取但未提交(回滚)的数据。 避免了“脏读取”和“不可重复读取”的情况,但是带来了更多的性能损失。

READ COMMITTED

⼤多数主流数据库的默认事务等级,保证了⼀个事务不会读到另⼀个并⾏事务 已修改但未提交的数据,避免了“脏读取”。该级别适⽤于⼤多数系统。

Read Uncommitted

保证了读取过程中不会读取到⾮法数据

我们来举个事务传播的例子:

假设有一个方法childTest(),它是一个事务方法,使用了PROPAGATION_REQUIRED事务传播选项。如果在一个事务中调用这个方法,那么它会在同一个事务中执行。

现在假设有一个方法mainTest(),它调用了childTest()方法。如果在mainTest()方法中没有显式地声明事务,而childTest()方法中声明了事务,那么childTest()方法将在一个新的事务中执行,这个新的事务不会影响mainTest()方法的执行。

但是,如果mainTest()方法也使用了事务传播选项,例如PROPAGATION_REQUIRED,那么在执行childTest()方法时,它的事务会隐式地与mainTest()方法的事务关联。这意味着如果childTest()方法的事务成功,则mainTest()方法的事务也会成功;如果childTest()方法的事务失败,则mainTest()方法的事务也会失败。

关于 Spring 的事务,你了解了么?





号外!号外!

Java 极客技术微信群中有很多优秀的小伙伴在讨论技术,偶尔还有不定期的资料分享和红包发放!如果你想提升自己,并且想和优秀的人一起进步,感兴趣的朋友,可以在下方公众号后台回复:加群

喜欢就分享
认同就点赞

支持就在看

一键四连,你的offer也四连



相关推荐

  • 香菜爱好者请进!大牌推出香菜口味新品?
  • DeepMind:谁说卷积网络不如ViT?
  • 35年首次证明!NYU重磅发现登Nature:神经网络具有类人泛化能力,举一反三超GPT-4
  • 面试太难,想要回去读研!
  • 竞赛这一年--从深度学习一窍不通到竞赛多个冠亚军。
  • MongoDB 的集群架构与设计
  • [开源]免费开源的流程设计器,开箱即用,助力流程开发使用更简单
  • 这几个编程原则,学校可不会教你
  • springboot第44集:Kafka集群和Lua脚本
  • 【ICCV2023】频率成分在少样本学习中的重要性
  • 北京大学王选所推出 ALCUNA ,大模型新知识理解能力评测再添一榜!
  • 重磅!GPT-4又进化了!画图、插件、代码等能力被整合,超级智能体来了
  • 深圳/北京内推 | 快手Y-tech部门招聘计算机视觉算法实习生
  • 博士申请 | 丹麦哥本哈根大学计算机系招收机器学习方向全奖博士生
  • NeurIPS 2023 | 连接多模态对比表征:无需配对数据的高效对比表征学习方法
  • 港大联合百度开源通用图大模型GraphGPT:让大语言模型读懂图数据
  • midjourney提示词下的皮克斯风格的不同地域情侣舞蹈
  • 推友展示ChatGpt新功能个如何把一些套壳和插件商直接拍死
  • 1029.AI日报:OpenAI推出了类似chatPDF的功能,再次碾压
  • 被客户投诉了,非要装XP系统