Spring注解驱动之后再说事务啊(spring注解驱动开发)
sinye56 2024-10-16 15:14 9 浏览 0 评论
问题引入
- Spring中事务传播有哪几种,分别是怎样的?
- 理解注解事务的自动配置?
- SpringBoot启动类为什么不需要加@EnableTransactionManagement注解?
- 声明式事务的实现原理?(待补充)
1 声明式事务
系统开发中必然与数据打交道,事务管理必不可少。Spring支持声明式事务,通过*@Transactional注解控制方法是否支持事务。声明式事务,基于AOP实现,将具体业务和业务逻辑解耦。
Spring提供了@EnableTransactionManagement注解在配置类(启动类)上启用支持事务,此时Spring会自动扫描具有@Transactional注解的类和方法。该注解相当于xml配置方式的 <tx:annotation-driven />。通过设置mode属性,决定使用spring代理,还是ASPECTJ扩展。
@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拦截器优先级都高于事务,所以被嵌套在里面,越贴近业务代码。
}
复制代码
2 @Transactional注解的使用
2.1 @Transactional注解属性
@Transactional注解可以应用于类和方法。声明类时,该注解默认作用于类和子类的所有方法,应用于public方法才有效;父类方法要加入同等级别的注解,需要单独声明。
@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
public @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使用而设计。
2.2 事务的传播行为-Propagation
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;
}
}
复制代码
- REQUIRED:使用当前的事务,如果当前没有事务,则自己新建一个事务,子方法是必须运行在一个事务中的;如果当前存在事务,则加入这个事务,成为一个整体。
举例:领导没饭吃,我有钱,那么我会自己买了吃;领导有的吃,会分给你一起吃。 - SUPPORTS:如果当前有事务,则使用事务;如果当前没有事务,则不使用事务。多用于查询。
举例:领导没饭吃,我也没饭吃;领导有饭吃,我也有饭吃。 - MANDATORY:传播属性强制必须存在一个事务,如果不存在,则会抛出异常。
举例:领导必须管饭,不管饭就没饭吃,我就不干了(就会抛出异常)。 - REQUIRES_NEW:如果当前有事务,则挂起该事务,并且自己创建一个新的事务给自己使用;如果当前没有事务,则同意 REQUIRED
举例:领导有饭吃,我偏不要,自己买东西自己吃 1.标志REQUIRES_NEW会新开启事务,外层事务不会影响内部事务的提交/回滚。内部提交修改,会导致A的脏读。 2.标志REQUIRES_NEW的内部事务异常,会影响外部事务的回滚 - NOT_SUPPORTED:如果当前有事务,则把事务挂起,自己不使用事务运行数据库操作
举例:领导有饭吃,分一点给你,我太忙了,放一边,我不吃 - NEVER:如果当前有事务存在,则抛出异常
举例:领导有饭给你吃,我不想吃,果断抛出异常 - NESTED:如果当前存在事务,则开启子事务(嵌套事务);如果当前没有事务,则同 REQUIRED。但是*如果主事务提交,则会携带子事务一起提交。如果主事务会回滚,则子事务会一起回滚。*相反,子事务异常,则父事务可以回滚或不回滚(trycatch)。
举例:领导决策不对,老板怪罪,领导带着小弟一同受罪;小弟出了差错,领导可以推卸责任。
区分NESTED与REQUIRES_NEW
最根本的区别是NESTED还在一个事务中,但是与主事务一块提交。
// TransactionalServiceImpl
@Transactional(propagation = Propagation.REQUIRED)
@Override
public void testPropagation() {
stuService.saveParent();
stuService.saveChildren();
int a = 1 / 0;
}
// StuServiceImpl
/* 测试事务传播 */
@Transactional(propagation = Propagation.NESTED) // 切换NESTED/REQUIRES_NEW
@Override
public void saveParent() {
Stu stu = new Stu();
stu.setName("parent");
stu.setScore(100);
stuMapper.insert(stu);
}
@Transactional(propagation = Propagation.NESTED)
@Override
public void saveChildren() {
saveChild1();
int a = 1 / 0;
saveChild2();
}
复制代码
一个容易疏漏的点
在代理模式(默认)下,仅可接通过代理传入的外部方法调用。这意味着同一个目标对象内部的方法调用,即使调用的方法标记有@Transactional,也不会在运行时导致事务拦截。
// 同一个类
@Transactional(propagation = Propagation.REQUIRED)
@Override
public 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。
复制代码
2.3 事务的隔离级别-Isolation
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种,隔离级别的设置依赖当前数据库是否支持。
- DEFAULT:使用当前数据库的默认隔离级别。例如Oracle是READ_COMMITED,MySQL是READ_REPEATED。
- READ_UNCOMMITED:可导致脏读、不可重复读、幻读。
- READ_COMMITTED:阻止脏读,可导致不可重复读、幻读。
- REPEATABLE_READ:阻止脏读和不可重复读,可导致幻读。
- SERIALIZABLE:该级别下事务顺序执行,阻止上面的缺陷,开销很大。
3 SpringBoot启动类为什么不需要加@EnableTransactionManagement注解
其实我自己很关注的是这个点,一般现在开发都是使用SpringBoot平台了。
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 {
}
}
复制代码
4 Spring声明式事务的内部实现机制
在应用系统调用声明@Transactional 的目标方法时,Spring Framework 默认使用 AOP 代理,结合AOP和事务元数据(注解)在代码运行时生成一个代理对象,根据@Transactional 的属性配置信息,这个代理对象决定该声明@Transactional 目标方法是否有拦截器 TransactionInterceptor 来使用拦截,在 TransactionInterceptor 拦截时,会在目标方法开始执行之前创建并加入事务,并执行目标方法的逻辑, 最后根据执行情况是否出现异常,利用抽象事务管理器AbstractPlatformTransactionManager 操作数据源 DataSource 提交或回滚事务。
事务管理的框架是由抽象事务管理器 AbstractPlatformTransactionManager 来提供的,而具体的底层事务处理实现由 PlatformTransactionManager 的具体实现类型来实现,如事务管理器 DataSourceTransactionManager。不同的事务管理器管理不同的数据资源 DataSource,比如 DataSourceTransactionManager 管理 JDBC 的 Connection。
相关推荐
- 程序员:JDK的安装与配置(完整版)_jdk的安装方法
-
对于Java程序员来说,jdk是必不陌生的一个词。但怎么安装配置jdk,对新手来说确实头疼的一件事情。我这里以jdk10为例,详细的说明讲解了jdk的安装和配置,如果有不明白的小伙伴可以评论区留言哦下...
- Linux中安装jdk并配置环境变量_linux jdk安装教程及环境变量配置
-
一、通过连接工具登录到Linux(我这里使用的Centos7.6版本)服务器连接工具有很多我就不一一介绍了今天使用比较常用的XShell工具登录成功如下:二、上传jdk安装包到Linux服务器jdk...
- 麒麟系统安装JAVA JDK教程_麒麟系统配置jdk
-
检查检查系统是否自带java在麒麟系统桌面空白处,右键“在终端打开”,打开shell对话框输入:java–version查看是否自带java及版本如图所示,系统自带OpenJDK,要先卸载自带JDK...
- 学习笔记-Linux JDK - 安装&配置
-
前提条件#检查是否存在JDKrpm-qa|grepjava#删除现存JDKyum-yremovejava*安装OracleJDK不分系统#进入安装文件目...
- Linux新手入门系列:Linux下jdk安装配置
-
本系列文章是把作者刚接触和学习Linux时候的实操记录分享出来,内容主要包括Linux入门的一些理论概念知识、Web程序、mysql数据库的简单安装部署,希望能够帮到一些初学者,少走一些弯路。注意:L...
- 测试员必备:Linux下安装JDK 1.8你必须知道的那些事
-
1.简介在Oracle收购Sun后,Java的一系列产品就被整合到Oracle官网中,打开官网乍眼一看也不知道去哪里下载,还得一个一个的摸索尝试,而且网上大多数都是一些Oracle收购Sun前,或者就...
- Linux 下安装JDK17_linux 安装jdk1.8 yum
-
一、安装环境操作系统:JDK版本:17二、安装步骤第一步:下载安装包下载Linux环境下的jdk1.8,请去官网(https://www.oracle.com/java/technologies/do...
- 在Ubuntu系统中安装JDK 17并配置环境变量教程
-
在Ubuntu系统上安装JDK17并配置环境变量是Java开发环境搭建的重要步骤。JDK17是Oracle提供的长期支持版本,广泛用于开发Java应用程序。以下是详细的步骤,帮助你在Ubuntu系...
- 如何在 Linux 上安装 Java_linux安装java的步骤
-
在桌面上拥抱Java应用程序,然后在所有桌面上运行它们。--SethKenlon(作者)无论你运行的是哪种操作系统,通常都有几种安装应用程序的方法。有时你可能会在应用程序商店中找到一个应用程序...
- Windows和Linux环境下的JDK安装教程
-
JavaDevelopmentKit(简称JDK),是Java开发的核心工具包,提供了Java应用程序的编译、运行和开发所需的各类工具和类库。它包括了JRE(JavaRuntimeEnviro...
- linux安装jdk_linux安装jdk软连接
-
JDK是啥就不用多介绍了哈,外行的人也不会进来看我的博文。依然记得读大学那会,第一次实验课就是在机房安装jdk,编写HelloWorld程序。时光飞逝啊,一下过了十多年了,挣了不少钱,买了跑车,娶了富...
- linux安装jdk,全局配置,不同用户不同jdk
-
jdk1.8安装包链接:https://pan.baidu.com/s/14qBrh6ZpLK04QS8ogCepwg提取码:09zs上传文件解压tar-zxvfjdk-8u152-linux-...
- 运维大神教你在linux下安装jdk8_linux安装jdk1.7
-
1.到官网下载适合自己机器的版本。楼主下载的是jdk-8u66-linux-i586.tar.gzhttp://www.oracle.com/technetwork/java/javase/downl...
- window和linux安装JDK1.8_linux 安装jdk1.8.tar
-
Windows安装JDK1.8的步骤:步骤1:下载JDK打开浏览器,找到JDK下载页面https://d.injdk.cn/download/oraclejdk/8在页面中找到并点击“下载...
- 最全的linux下安装JavaJDK的教程(图文详解)不会安装你来打我?
-
默认已经有了linux服务器,且有root账号首先检查一下是否已经安装过java的jdk任意位置输入命令:whichjava像我这个已经安装过了,就会提示在哪个位置,你的肯定是找不到。一般我们在...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- oracle忘记用户名密码 (59)
- oracle11gr2安装教程 (55)
- mybatis调用oracle存储过程 (67)
- oracle spool的用法 (57)
- oracle asm 磁盘管理 (67)
- 前端 设计模式 (64)
- 前端面试vue (56)
- linux格式化 (55)
- linux图形界面 (62)
- linux文件压缩 (75)
- Linux设置权限 (53)
- linux服务器配置 (62)
- mysql安装linux (71)
- linux启动命令 (59)
- 查看linux磁盘 (72)
- linux用户组 (74)
- linux多线程 (70)
- linux设备驱动 (53)
- linux自启动 (59)
- linux网络命令 (55)
- linux传文件 (60)
- linux打包文件 (58)
- linux查看数据库 (61)
- linux获取ip (64)
- linux进程通信 (63)