摘要:支持声明式事务,通过注解控制方法是否支持事务。声明式事务,基于实现,将具体业务和业务逻辑解耦。该级别下事务顺序执行,阻止上面的缺陷,开销很大。
问题引入
- Spring中事务传播有哪几种,分别是怎样的?
- 理解注解事务的自动配置?
- SpringBoot启动类为什么不需要加@EnableTransactionManagement注解?
- 声明式事务的实现原理?(待补充)
系统开发中必然与数据打交道,事务管理必不可少。Spring支持声明式事务,通过@Transactional注解控制方法是否支持事务。声明式事务,基于AOP实现,将具体业务和业务逻辑解耦。
Spring提供了@EnableTransactionManagement注解在配置类(启动类)上启用支持事务,此时Spring会自动扫描具有@Transactional注解的类和方法。该注解相当于xml配置方式的
@Target(ElementType.TYPE)@Retention(RetentionPolicy.RUNTIME)@Documented@Import(TransactionManagementConfigurationSelector.class)public @interface EnableTransactionManagement { boolean proxyTargetClass() default false; AdviceMode mode() default AdviceMode.PROXY; // 代理模式 int order() default Ordered.LOWEST_PRECEDENCE; // LOWEST_PRECEDENCE最低优先级,所以在执行链的最外面,而自己实现的AOP拦截器优先级都高于事务,所以被嵌套在里面,越贴近业务代码。}
@Transactional注解可以应用于类和方法。声明类时,该注解默认作用于类和子类的所有方法,应用于public方法才有效;父类方法要加入同等级别的注解,需要多带带声明。
@Target({ElementType.TYPE, ElementType.METHOD})@Retention(RetentionPolicy.RUNTIME)@Inherited@Documentedpublic @interface Transactional { @AliasFor("transactionManager") String value() default ""; // 用来确定目标事务管理器 @AliasFor("value") String transactionManager() default ""; // 事务的传播,默认Propagation.REQUIRED Propagation propagation() default Propagation.REQUIRED; // 事务隔离级别,默认是Isolation.DEFAULT Isolation isolation() default Isolation.DEFAULT; // 事务超时时间,默认是-1 int timeout() default TransactionDefinition.TIMEOUT_DEFAULT; // 指定事务是否为只读事务,默认为false,仅仅是个提示 boolean readOnly() default false; // 标识能触发事务回滚的异常类型,默认是RuntimeException和Error,不包含检查异常。 Class extends Throwable>[] rollbackFor() default {}; // 标识哪些异常不需要回滚事务 Class extends Throwable>[] noRollbackFor() default {}; String[] noRollbackForClassName() default {}; String[] rollbackForClassName() default {};}
其中,isolation和timeout两个属性仅对新启动的事务有效,专为Propagation.REQUIRED和Propagation.REQUIRES_NEW使用而设计。
Propagation定义了事务的传播,一共有7种级别。
public enum Propagation { REQUIRED(TransactionDefinition.PROPAGATION_REQUIRED), SUPPORTS(TransactionDefinition.PROPAGATION_SUPPORTS), MANDATORY(TransactionDefinition.PROPAGATION_MANDATORY), REQUIRES_NEW(TransactionDefinition.PROPAGATION_REQUIRES_NEW), NOT_SUPPORTED(TransactionDefinition.PROPAGATION_NOT_SUPPORTED), NEVER(TransactionDefinition.PROPAGATION_NEVER), NESTED(TransactionDefinition.PROPAGATION_NESTED); private final int value; Propagation(int value) { this.value = value; } public int value() { return this.value; }}
最根本的区别是NESTED还在一个事务中,但是与主事务一块提交。
// TransactionalServiceImpl@Transactional(propagation = Propagation.REQUIRED)@Overridepublic void testPropagation() { stuService.saveParent(); stuService.saveChildren(); int a = 1 / 0;}// StuServiceImpl/* 测试事务传播 */@Transactional(propagation = Propagation.NESTED) // 切换NESTED/REQUIRES_NEW@Overridepublic void saveParent() { Stu stu = new Stu(); stu.setName("parent"); stu.setScore(100); stuMapper.insert(stu);}@Transactional(propagation = Propagation.NESTED)@Overridepublic void saveChildren() { saveChild1(); int a = 1 / 0; saveChild2();}
在代理模式(默认)下,仅拦截通过代理传入的外部方法调用。这意味着同一个目标对象内部的方法调用,即使调用的方法标记有@Transactional,也不会在运行时导致事务拦截。
// 同一个类@Transactional(propagation = Propagation.REQUIRED)@Overridepublic void saveChildren() { saveChild1(); int a = 1 / 0; saveChild2();}@Transactional(propagation = Propagation.REQUIRES_NEW)public void saveChild1() { Stu stu = new Stu(); stu.setName("child-1"); stu.setScore(60); stuMapper.insert(stu);}
参考官方文档
In proxy mode (which is the default), only external method calls coming in through the proxy are intercepted. This means that self-invocation, in effect, a method within the target object calling another method of the target object, will not lead to an actual transaction at runtime even if the invoked method is marked with @Transactional. Also, the proxy must be fully initialized to provide the expected behaviour so you should not rely on this feature in your initialization code, i.e. @PostConstruct.在代理模式下(默认),只有通过代理进入的外部方法调用才会被拦截。 这意味着自调用实际上是目标对象中的一个方法调用目标对象的另一个方法,即使被调用的方法用@Transactional 标记,也不会在运行时导致实际事务。 此外,代理必须完全初始化以提供预期的行为,因此您不应在初始化代码中依赖此功能,即@PostConstruct。
public enum Isolation { DEFAULT(TransactionDefinition.ISOLATION_DEFAULT), READ_UNCOMMITTED(TransactionDefinition.ISOLATION_READ_UNCOMMITTED), READ_COMMITTED(TransactionDefinition.ISOLATION_READ_COMMITTED), REPEATABLE_READ(TransactionDefinition.ISOLATION_REPEATABLE_READ), SERIALIZABLE(TransactionDefinition.ISOLATION_SERIALIZABLE); private final int value; Isolation(int value) { this.value = value; } public int value() { return this.value; }}
Spring事务隔离级别共有5种,隔离级别的设置依赖当前数据库是否支持。
org.springframework.boot.autoconfigure.transaction.TransactionAutoConfiguration
SpringBoot加载spring.factories时,会加载事务配置类TransactionAutoConfiguration,内部有开启事务管理的配置。
// ~TransactionAutoConfiguration中的内部类@Configuration(proxyBeanMethods = false)@ConditionalOnBean(TransactionManager.class)@ConditionalOnMissingBean(AbstractTransactionManagementConfiguration.class)public static class EnableTransactionManagementConfiguration { @Configuration(proxyBeanMethods = false) @EnableTransactionManagement(proxyTargetClass = false) @ConditionalOnProperty(prefix = "spring.aop", name = "proxy-target-class", havingValue = "false", matchIfMissing = false) public static class JdkDynamicAutoProxyConfiguration { } @Configuration(proxyBeanMethods = false) @EnableTransactionManagement(proxyTargetClass = true) @ConditionalOnProperty(prefix = "spring.aop", name = "proxy-target-class", havingValue = "true", matchIfMissing = true) public static class CglibAutoProxyConfiguration { }}
在应用系统调用声明@Transactional 的目标方法时,Spring Framework 默认使用 AOP 代理,结合AOP和事务元数据(注解)在代码运行时生成一个代理对象,根据@Transactional 的属性配置信息,这个代理对象决定该声明@Transactional 的目标方法是否由拦截器 TransactionInterceptor 来使用拦截,在 TransactionInterceptor 拦截时,会在目标方法开始执行之前创建并加入事务,并执行目标方法的逻辑, 最后根据执行情况是否出现异常,利用抽象事务管理器AbstractPlatformTransactionManager 操作数据源 DataSource 提交或回滚事务。
事务管理的框架是由抽象事务管理器 AbstractPlatformTransactionManager 来提供的,而具体的底层事务处理实现由 PlatformTransactionManager 的具体实现类来实现,如事务管理器 DataSourceTransactionManager。不同的事务管理器管理不同的数据资源 DataSource,比如 DataSourceTransactionManager 管理 JDBC 的 Connection。
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/124120.html
摘要:基于和命名空间的声明式事务管理目前推荐的方式,其最大特点是与结合紧密,可以充分利用切点表达式的强大支持,使得管理事务更加灵活。基于的全注解方式将声明式事务管理简化到了极致。 Java面试通关手册(Java学习指南):https://github.com/Snailclimb/Java_Guide 历史回顾:可能是最漂亮的Spring事务管理详解 Spring事务管理 Spring支持两...
摘要:当你真正到公司里面从事了几年开发之后,你就会同意我的说法利用找工作,需要的就是项目经验,项目经验就是理解项目开发的基本过程,理解项目的分析方法,理解项目的设计思 Java就是用来做项目的!Java的主要应用领域就是企业级的项目开发!要想从事企业级的项目开发,你必须掌握如下要点: 1、掌握项目开发的基本步骤 2、具备极强的面向对象的分析与设计技巧 3、掌握用例驱动、以架构为核心的主流开发...
摘要:前言如题,今天介绍的声明式事务。提供一个注解在配置类上来开启声明式事务的支持。而在配置里还开启了对声明式事务的支持,代码如下所以在中,无须显式开启使用注解。源码下载后语以上为声明式事务的教程。 微信公众号:一个优秀的废人如有问题或建议,请后台留言,我会尽力解决你的问题。 前言 如题,今天介绍 SpringBoot 的 声明式事务。 Spring 的事务机制 所有的数据访问技术都有事务处...
阅读 945·2021-11-22 12:09
阅读 3713·2021-09-27 13:36
阅读 1400·2021-08-20 09:37
阅读 4025·2019-12-27 12:22
阅读 2363·2019-08-30 15:55
阅读 2370·2019-08-30 13:16
阅读 2829·2019-08-26 17:06
阅读 3441·2019-08-23 18:32