注解@Transactional(rollbackFor = Exception.class) 的用法

本文探讨了在Spring框架下如何正确使用事务管理,特别是在循环中处理多个数据库操作时,如何确保部分失败不会影响整体事务的正确性。通过将业务逻辑放入独立的服务类并正确应用@Transactional注解,实现了细粒度的事务控制。

由于业务需求要求,在一个Service的一个方法A中有一个for循环,每次循环里面的业务逻辑有可能发生异常,这个时候就需要将这个循环内的所有数据库操作给回滚掉,但是又不能影响到之前循环里数据的更改,并且后面的循环里不发生异常的情况下也需要正常操作数据库

for (int i = 0; i < educationInfoArrayList.size(); i++) {
    String result = insertOrUpdateByOnce(educationInfoArrayList.get(i), beforeEntryArrayList.get(i), afterEntryArrayList.get(i), educationExperienceArrayList.get(i),
            assessInfoArrayList.get(i), qualificationClassificationArrayList.get(i), trainingInfoArrayList.get(i));
}

@Transactional(rollbackFor = Exception.class)
public String insertOrUpdateByOnce(EducationInfo educationInfo, BeforeEntry beforeEntry, AfterEntry afterEntry, EducationExperience educationExperience, AssessInfo assessInfo,
        QualificationClassification qualificationClassification, TrainingInfo trainingInfo) {

    try {
        //这里写核心业务方法
    } catch (Exception e) {
        log.error("插入或者修改异常");
        //失败则手动回滚
        TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
        //返回异常信息
        return e.getCause().getMessage();
    }

    return "success";

}

catch中加上手动回滚 TransactionAspectSupport.currentTransactionStatus().setRollbackOnly(),事务也没有生效,且报错

org.springframework.transaction.NoTransactionException: No transaction aspect-managed TransactionStatus in scope
    at org.springframework.transaction.interceptor.TransactionAspectSupport.currentTransactionStatus(TransactionAspectSupport.java:122)
    at com.dubboconsumer.dubboConsumer.service.imp.DemoService.B(DemoService.java:44)
    at com.dubboconsumer.dubboConsumer.service.imp.DemoService.A(DemoService.java:20)
    at com.dubboconsumer.dubboConsumer.service.imp.DemoService$$FastClassBySpringCGLIB$$c7449c96.invoke(<generated>)
    at org.springframework.cglib.proxy.MethodProxy.invoke(MethodProxy.java:204)
    at org.springframework.aop.framework.CglibAopProxy$DynamicAdvisedInterceptor.intercept(CglibAopProxy.java:669)
    ...

原因是:调用了同一个类的方法,给这个方法配上事务注解是不会生效的,这个类里面没有事务,不能进行手动回滚;Spring之所以可以对开启@Transactional的方法进行事务管理,是因为Spring为当前类生成了一个代理类,然后在执行相关方法时,会判断这个方法有没有@Transactional注解,如果有的话,则会开启一个事务;所以,要把执行事务的方法全部丢到一个service类,在这个service类上配上注解@Transactional(rollbackFor = Exception.class),则这个service类中所有的方法都是执行事务的管理

 

for (int i = 0; i < educationInfoArrayList.size(); i++) {
    String result = insertOrUpdateEmployeeService.insertOrUpdateByOnce(educationInfoArrayList.get(i), beforeEntryArrayList.get(i), afterEntryArrayList.get(i), educationExperienceArrayList.get(i),
                    assessInfoArrayList.get(i), qualificationClassificationArrayList.get(i), trainingInfoArrayList.get(i));
}

 

@Repository
@Transactional(rollbackFor = Exception.class)
public class InsertOrUpdateEmployeeService {

    private static final Logger log = LoggerFactory.getLogger(InsertOrUpdateEmployeeService.class);

    @Autowired
    private EmployeeDao employeeDao;

    public String insertOrUpdateByOnce(EducationInfo educationInfo, BeforeEntry beforeEntry, AfterEntry afterEntry, EducationExperience educationExperience, AssessInfo assessInfo,
                                       QualificationClassification qualificationClassification, TrainingInfo trainingInfo) {

        try {
            //这里写核心业务方法
        } catch (Exception e) {
            log.error("插入或者修改异常");
            //失败则手动回滚
            TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
            //返回异常信息
            return e.getCause().getMessage();
        }

        return "success";

    }
}

这样就符合事务管理的要求了

转载于:https://www.cnblogs.com/liuqing576598117/p/10783821.html

### 区别 `@Transactional` 是 Spring 框架用于声明式事务管理的注解,默认情况下,仅对 `RuntimeException` 及其子类和 `Error` 类型的异常进行事务回滚操作。而 `@Transactional(rollbackFor = Exception.class)` 指定了当事务方法抛出 `Exception` 及其子类异常时进行回滚,将回滚范围扩大到所有异常类型,包括检查异常(如 `IOException` 等需要在方法签名中声明抛出的异常)[^3][^4][^5]。 ### 使用场景 - **`@Transactional`**:适用于业务逻辑中主要关注运行时异常和错误的情况。当业务代码抛出的异常大多是运行时异常,且不需要对受检查异常进行回滚处理时,使用该注解即可。例如,在一个简单的用户查询服务中,可能会因为数据库连接问题抛出 `RuntimeException` 及其子类异常,此时使用 `@Transactional` 可以保证事务的正确回滚。 ```java import org.springframework.transaction.annotation.Transactional; @Transactional public class UserQueryService { public User getUserById(Long id) { // 查询用户的业务逻辑 return null; } } ``` - **`@Transactional(rollbackFor = Exception.class)`**:当业务逻辑中可能会抛出受检查异常,并且希望在这些异常发生时也能回滚事务时,需要使用该注解。例如,在文件上传并保存文件信息到数据库的业务中,可能会抛出 `IOException` 等受检查异常,使用 `@Transactional(rollbackFor = Exception.class)` 能确保在发生任何异常时事务都能回滚。 ```java import org.springframework.transaction.annotation.Transactional; import java.io.IOException; @Transactional(rollbackFor = Exception.class) public class FileUploadService { public void uploadFile(File file) throws Exception { // 上传文件并保存文件信息到数据库的业务逻辑 } } ``` ### 作用 两者的主要作用都是实现事务管理,保证一系列数据库操作的原子性、一致性、隔离性和持久性。`@Transactional` 提供了基本的事务管理功能,默认对运行时异常和错误进行回滚;而 `@Transactional(rollbackFor = Exception.class)` 则进一步增强了事务管理的能力,能够对所有类型的异常进行回滚,从而更好地保证数据的完整性和一致性[^1][^4]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值