springboot 事务手动回滚_SpringBoot事务简单操作及手动回滚

本文介绍了SpringBoot中事务管理的使用,包括如何手动回滚事务,设置回滚点以及使用DataSourceTransactionManager进行事务操作。在遇到错误时,通过@Transactional(rollbackFor = Exception.class)进行事务回滚,并展示了在特定异常情况下如何部分回滚事务。同时,文章还提到了Spring Boot controller中@Transactional不回滚的解决策略。
摘要由CSDN通过智能技术生成

一、引入依赖

org.springframework.boot

spring-boot-starter

org.springframework.boot

spring-boot-starter-data-jdbc

mysql

mysql-connector-java

5.1.48

org.springframework.boot

spring-boot-starter-test

二、application.properties

spring.datasource.url=jdbc:mysql://192.168.178.5:12345/mydb?characterEncoding=UTF-8spring.datasource.username=root

spring.datasource.password=123456spring.datasource.driver-class-name=com.mysql.jdbc.Driver

spring.datasource.type=com.zaxxer.hikari.HikariDataSource

三、dao和service代码

1. dao

student接口

public interfaceStudentDao {voidsaveStudent();

}

student实现类

@Repositorypublic class StudentDaoImpl implementsStudentDao{

@AutowiredprivateJdbcTemplate jdbcTemplate;public voidsaveStudent(){

String sql= "insert into student(name, s_class) values(?,?"; //这里会出现一个错误,少了右括号

List sqlParamList = new ArrayList();

sqlParamList.add("stn-"+Math.random());

sqlParamList.add(20);

jdbcTemplate.update(sql, sqlParamList.toArray(newObject[sqlParamList.size()]));

}

}

user接口

public interfaceUserDao {voidsaveUser();

}

user实现类

@Repositorypublic class UserDaoImpl implementsUserDao{

@AutowiredprivateJdbcTemplate jdbcTemplate;public voidsaveUser(){

String sql= "insert into user(name, age) values(?,?)";

List sqlParamList = new ArrayList();

sqlParamList.add("sn-"+Math.random());

sqlParamList.add(20);

jdbcTemplate.update(sql, sqlParamList.toArray(newObject[sqlParamList.size()]));

}

}

2. service

接口:

public interfaceOperatorService {void saveEntity() throwsException;

}

接口实现类

@Servicepublic class OperatorServiceImpl implementsOperatorService{

@AutowiredprivateStudentDao studentDao;

@AutowiredprivateUserDao userDao;

@Overridepublic void saveEntity() throwsException{

userDao.saveUser();

studentDao.saveStudent();

}

}

当执行以上saveEntity()代码时,因StudentDaoImpl 插入语句的一个错误,会导致事务不一致,user表成功插入一条记录,student表没有。

为了使事务一致,在SpringBoot项目中,我们只需要在saveEntity()上添加@Transactional注解,,对@Transactional的注解可以查看 对注解@Transactional的解读 一节。

这里为了适应更多的异常,我们提升了事务捕获异常的范围:@Transactional(rollbackFor = Exception.class)

四、手动回滚事务

有时我们需要捕获一些错误信息,又需要进行事务回滚,这时我们就需要用到Spring提供的事务切面支持类TransactionAspectSupport。

@Transactional(rollbackFor = Exception.class)

@Overridepublic void saveEntity() throwsException{try{

userDao.saveUser();

studentDao.saveStudent();

}catch(Exception e){

System.out.println("异常了=====" +e);//手动强制回滚事务,这里一定要第一时间处理

TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();

}

}

手动回滚事务一定要加上@Transactional,不然会报以下错误:

org.springframework.transaction.NoTransactionException: No transaction aspect-managed TransactionStatus in scope

想想也是,不开启事务,何来手动回滚,所以@Transactional必不可少。

五、回滚部分异常

使用Object savePoint = TransactionAspectSupport.currentTransactionStatus().createSavepoint(); 设置回滚点。

使用TransactionAspectSupport.currentTransactionStatus().rollbackToSavepoint(savePoint); 回滚到savePoint。

@Transactional(rollbackFor = Exception.class)

@Overridepublic void saveEntity() throwsException{

Object savePoint= null;try{

userDao.saveUser();//设置回滚点

savePoint =TransactionAspectSupport.currentTransactionStatus().createSavepoint();

studentDao.saveStudent();//执行成功

int a = 10/0; //这里因为除数0会报异常,进入catch块

}catch(Exception e){

System.out.println("异常了=====" +e);//手工回滚异常

TransactionAspectSupport.currentTransactionStatus().rollbackToSavepoint(savePoint);

}

}

六、使用DataSourceTransactionManager

springboot 开启事务以及手动提交事务,可以在服务类上加上两个注解。

@Autowired

DataSourceTransactionManager dataSourceTransactionManager;

@Autowired

TransactionDefinition transactionDefinition;

手动开启事务

TransactionStatus transactionStatus = dataSourceTransactionManager.getTransaction(transactionDefinition);

手动提交事务

dataSourceTransactionManager.commit(transactionStatus);//提交

手动回滚事务

dataSourceTransactionManager.rollback(transactionStatus);//最好是放在catch 里面,防止程序异常而事务一直卡在哪里未提交

七、spring boot controller设置 @Transactional 不回滚的解决办法

默认spring事务只在发生未被捕获的 RuntimeException 时才回滚。

spring aop  异常捕获原理:被拦截的方法需显式抛出异常,并不能经任何处理,这样aop代理才能捕获到方法的异常,才能进行回滚,默认情况下aop只捕获 RuntimeException 的异常,但可以通过配置来捕获特定的异常并回滚。

换句话说在service的方法中不使用try catch 或者在catch中最后加上throw new RuntimeException (),这样程序异常时才能被aop捕获进而回滚。

解决方案:

方案1:例如service层处理事务,那么service中的方法中不做异常捕获,或者在catch语句中最后增加throw new RuntimeException()语句,以便让aop捕获异常再去回滚,并且在service上层(webservice客户端,view层action)要继续捕获这个异常并处理。

方案2:在service层方法的catch语句中增加:TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();语句,手动回滚,这样上层就无需去处理异常。

注意:

1. 默认地,如果使用的数据源不是SpringBoot的默认配置(即是由自己定义的配置信息,自己解析创建的数据源),则需要手动创建事务管理器,因为SpringBoot无法识别配置信息,无法完成自动注入。

//DynamicDataSource 是自定义的数据源

@BeanpublicPlatformTransactionManager transactionManager(DynamicDataSource dataSource){return newDataSourceTransactionManager(dataSource);

}

2. SpringBoot1.x需要在启动类上添加@EnableTransactionManagement,SpringBoot2.x则不需要。

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值