Java Spring AOP源码解析之事务实现原理

网友投稿 241 2022-09-05


Java Spring AOP源码解析之事务实现原理

目录不用Spring管理事务?编程式事务管理使用PlatformTransactionManager使用TransactionTemplate声明式事务管理使用@Transactional注解源码解析参考博客总结

不用Spring管理事务?

让我们先来看一下不用spring管理事务时,各种框架是如何管理事务的

使用JDBC来管理事务

使用Hibernate来管理事务

业务逻辑和事务代码是耦合到一块的,并且和框架的具体api绑定了。当我们换一种框架来实现时,里面对事务控制的代码就要推倒重写,并不一定能保证替换后的api和之前的api有相同的行为。

基于这些问题,Spring抽象了一些事务相关的顶层接口,我们面向接口编程,换框架时只要换具体的实现即可。有点像JDBC API的味道了

常用api接口PlatformTransactionManager对事务进行管理TransactionDefinition定义事务的相关属性,例如隔离级别,传播行为TransactionStatus保存事务状态

针对不同的数据访问技术,使用不用的PlatformTransactionManager类即可

数据访问技术PlatformTransactionManager实现类JDBC/MybatisDataSourceTransactionManagerHibernateHibernateTransactionManagerJpaJpaTransactionManagerJmsJmsTransactionManager

编程式事务管理

使用PlatformTransactionManager

使用TransactionTemplate

当我们直接使用PlatformTransactionManager来管理事务时,有很多模版代码。例如业务代码正常执行,提交事务,否则回滚事务。我们可以把这部分模版代码封装成一个模版类,这样使用起来就很方便了,如下所示

如下图所示,TransactionTemplate#execute方法就是一个典型的模版方法

我们可以传入如下2个接口的实现类来执行业务逻辑,TransactionCallback(需要返回执行结果)或TransactionCallbackWithoutResult(不需要返回结果)

声明式事务管理

为了让使用更加简洁,Spring直接把事务代码的执行放到切面中了,我们只需要在业务代码方法上加上一个@Transactional注解即可,这种方式我们最常用哈

使用@Transactional注解

此时事务相关的定义我们就可以通过@Transactional注解来设置了

属性名类型描述默认值value(和transactionManager互为别名)String当在配置文件中有多个PlatformTransactionManager ,用该属性指定选择哪个事务管理器空字符串""propagation枚举:Propagation事务的传播行为REQUIREDisolation枚举:Isolation事务的隔离度DEFAULTtimeoutint事务的超时时间。如果超过该时间限制但事务还没有完成,则自动回滚事务-1readOnlyboolean指定事务是否为只读事务falserollbackForClass[]需http://要回滚的异常空数组{}rollbackForClassNameString[]需要回滚的异常类名空数组{}noRollbackForClass[]不需要回滚的异常空数组{}noRollbackForClassNameString[]不需要回滚的异常类名空数组{}

源码解析

我们需要在配置类上加上@EnableTransactionManagement注解,来开启spring事务管理功能,@EnableTransactionManagement最主要的功能就是注入一个TransactionInterceptor拦截器,来控制事务开启,提交或者回滚

ProxyTransactionManagementConfiguration

TransactionInterceptor#invoke

TransactionAspectSupport#invokeWithinTransaction

TransactionAspectSupport#createTransactionIfNecessary

当开启事务的时候,可以看到各种传播属性的行为

AbstractPlatformTransactionManager#getTransaction

Spring事务的传播行为在Propagation枚举类中定义了如下几种选择

支持当前事务

REQUIRED :如果当前存在事务,则加入该事务。如果当前没有事务,则创建一个新的事务SUPPORTS:如果当前存在事务,则加入该事务 。如果当前没有事务, 则以非事务的方式继续运行MANDATORY :如果当前存在事务,则加入该事务 。如果当前没有事务,则抛出异常

不支持当前事务

REQUIRES_NEW :创建一个新事务,如果当前存在事务,则把当前事务挂起NOT_SUPPORTED : 以非事务方式运行,如果当前存在事务,则把当前事务挂起NEVER : 以非事务方式运行,如果当前存在事务,则抛出异常

其他情况

NESTED :如果当前存在事务,则创建一个事务作为当前事务的嵌套事务来执行 。如果当前没有事务,则该取值等价于REQUIRED

以NESTED启动的事务内嵌于外部事务中 (如果存在外部事务的话),此时内嵌事务并不是一个独立的事务,它依赖于外部事务。只有通过外部事务的提交,才能引起内部事务的提交,嵌套的子事务不能单独提交

参考博客

https://jb51.net/article/229456.htm

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注我们的更多内容!


版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:【SVM预测】基于松鼠算法优化支持向量机SVM实现数据预测附matlab代码
下一篇:【成熟度检测】基于RGB和HSV实现水果成熟度分级系统含Matlab源码
相关文章

 发表评论

暂时没有评论,来抢沙发吧~