Spring事务的本质就是对数据库事务的支持,没有数据库事务,Spring是无法提供事务功能的。Spring只提供统一的事务管理接口,具体实现都是由数据库自己实现的,Spring会在事务开始时,根据当前设置的隔离级别,调整数据库的隔离级别,由此保持
为什么需要事务
事务是将一组操作封装成一个执行单元,要么全部成功,要么全部失败。如果没有事务,转账操作就会出现异常,因此需要保证原子性。
Spring 声明事务
- 只需要在方法上添加@Transactional注解就可以实现,无需手动开启和提交事务,如果中途发生异常就会自动回滚事务。
- @Transactional既可以修饰方法也可以修饰类,修饰方法只能修饰public方法
- 注意:@Transactional在异常被捕获的情况下,事务就不会自动回滚
@Transactional(isolation = Isolation.READ_COMMITTED) // 开启事务
@RequestMapping("/insert")
public int insert(){
UserInfo userInfo = new UserInfo();
userInfo.setName("AOP");
userInfo.setPassword("123456");
return userService.add(userInfo);
}
@Transactional(isolation = Isolation.READ_COMMITTED) // 开启事务
@RequestMapping("/insert2")
public int insert2(){
UserInfo userInfo = new UserInfo();
userInfo.setName("AOP");
userInfo.setPassword("123456");
int result = userService.add(userInfo);
System.out.println("MySQL影响的行数:"+result);
int num = 10/0;
return result;
}
Transactional参数说明
propagation
事务的传播行为,默认为Propagation.REQUIRED
isolation
Spring有5种隔离级别,MySQL有四种隔离级别,Spring多了一个DEFAULT,以连接的数据库的事务隔离级别为主
timeout
表示事务的超时时间,默认值为-1,如果事务执行时间超过超时时间,就会自动回滚
事务回滚失效解决方案
直接抛出异常
@Transactional(propagation = Propagation.REQUIRED) // 开启事务
@RequestMapping("/insert2")
public int insert2(){
UserInfo userInfo = new UserInfo();
userInfo.setName("AOP");
userInfo.setPassword("123456");
int result = userService.add(userInfo);
System.out.println("MySQL影响的行数:"+result);
try {
int num = 10/0;
}catch (Exception e){
result = 0;
e.printStackTrace();
throw e;//解决事务失效的问题
// TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
}
return result;
}
在catch中使用代码手动进行事务的回滚操作
@Transactional(propagation = Propagation.REQUIRED) // 开启事务
@RequestMapping("/insert2")
public int insert2(){
UserInfo userInfo = new UserInfo();
userInfo.setName("AOP");
userInfo.setPassword("123456");
int result = userService.add(userInfo);
System.out.println("MySQL影响的行数:"+result);
try {
int num = 10/0;
}catch (Exception e){
result = 0;
e.printStackTrace();
//throw e;//解决事务失效的问题
TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
}
return result;
}
@Transactional工作原理
@Transactional是基于AOP实现的,AOP又是使用动态代理实现的。如果目标对象实现了接口,默认情况下会采用JDK的动态代理,如果目标对象没有实现了接口,会使用CGLIB动态代理。
@Transactional在开始执行业务之前,通过代理先开启事务,在执行成功之后再提交事务。如果中途遇到了异常,则回滚事务。
Spring 事务的传播机制
定义:Spring中多个事务相互调用时,他们之间的行为方式是如何执行的
为什么需要事务传播机制?
事务的隔离级别是保证多个并发事务执行的可控性(稳定性),而事务传播机制是保证一个事务在多个调用方法间的可控性
传播机制的类型
嵌套事务和加入事务的区别
- 整个事务如果全部执行成功,二者的结果是一样的
- 如果事务执行到一半失败了,那么加入事务整个事务会全部回滚;而嵌套事务会局部回滚,不会影响上一个方法中执行的结果
到此这篇关于Spring超详细讲解事务和事务传播机制的文章就介绍到这了,更多相关Spring事务内容请搜索编程学习网以前的文章希望大家以后多多支持编程学习网!
本文标题为:Spring超详细讲解事务和事务传播机制
- ExecutorService Callable Future多线程返回结果原理解析 2023-06-01
- Spring Security权限想要细化到按钮实现示例 2023-03-07
- Springboot整合minio实现文件服务的教程详解 2022-12-03
- Java中的日期时间处理及格式化处理 2023-04-18
- SpringBoot使用thymeleaf实现一个前端表格方法详解 2023-06-06
- JSP页面间传值问题实例简析 2023-08-03
- 深入了解Spring的事务传播机制 2023-06-02
- Java实现顺序表的操作详解 2023-05-19
- JSP 制作验证码的实例详解 2023-07-30
- 基于Java Agent的premain方式实现方法耗时监控问题 2023-06-17