Spring MVC @Transactional注解方式事务无法回滚原因及解决方案

近期在做一个接口,采用的是Spring MVC的框架写的,但是当配置好框架,写完方法之后,发现在service层加了@Transactional注解的方式,竟然无效,无法进行事务的回滚,发现一般问题都不存在,一般@Transactional注解了,事务无法回滚是因为三个原因:

1.方法可能不是public的声明

2.异常类型是不是unchecked异常

这个说明一下,在@Transactional注解事务的特性的时候,查阅文档可以知道,@Transactional注解事务的话,默认是Spring会对unchecked异常进行事务回滚;如果是checked异常则不回滚。什么是unchecked?

java里面将派生于Error或者RuntimeException(比如空指针,1/0)的异常称为unchecked异常,其他继承自java.lang.Exception得异常统称为Checked Exception,如IOException、TimeoutException等

当然如需要让checked类型的也进行事务的回滚,可以照以下写法:

@Transactional(rollbackFor=Exception.class) 

3.数据库引擎要支持事务,如果是MySQL,注意表要使用支持事务的引擎,比如innodb,如果是myisam,事务是不起作用的

4.是否开启了对注解的解析

<tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true"/>

5.spring是否扫描到你这个包,如下是扫描到org.test下面的包

<context:component-scan base-package="org.test" ></context:component-scan>

6.检查是不是同一个类中的方法调用(如a方法调用同一个类中的b方法) 

如果是同一个类中的方法调用,那么需要两个方法都加@Transactional注解,这样才会被@Transactional认为是同一个事务管理


7.异常是不是被你catch住了

如果在@Transactional注解的方法中,你写了一个try/catch,那么在catch中你不写throw new Exception将异常抛出的话,是无法进行事务回滚的

@Transactional(rollbackFor = { Exception.class })    
public void test() {    
     try {    
        doDbStuff1();    
        doDbStuff2();    
     } catch (Exception e) {    
          e.printStackTrace();       
          //在这里如果不抛出异常,或者不手动回滚事物,那么是不会生效的
     }    
}  

解决方法:

@Transactional(rollbackFor = { Exception.class })    
public void test() {    
     try {    
        test1();    
        test2();    
     } catch (Exception e) {    
          e.printStackTrace();       
          //在这里可以抛出异常或者代码控制回滚
            解决方法一、
            throw new Exception("抛出异常");
            解决方法二、
            TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
            //增加这句话,就是用代码控制事务回滚
     }    
}  

以上问题,博主都没有无法解决自己的事务回滚问题,查了许多资料,才发现还有一个问题,就是在配置的时候:

在 applicationContext.xml配置扫描之后

<context:component-scan base-package=“com.will”>   
</context:component-scan>  

在 Spring mvc.dispatcher.xml的文件中需要增加

<context:component-scan base-package=“com.will” >      
    <context:exclude-filter type=“annotation” expression=“org.springframework.stereotype.Service” />    
</context:component-scan>  

Spring容器优先加载由ServletContextListener(对应applicationContext.xml)产生的父容器,而SpringMVC(对应mvc_dispatcher_servlet.xml)产生的是子容器。子容器Controller进行扫描装配时装配的@Service注解的实例是没有经过事务加强处理,即没有事务处理能力的Service,而父容器进行初始化的Service是保证事务的增强处理能力的。如果不在子容器中将Service exclude掉,此时得到的将是原样的无事务处理能力的Service。

  • 3
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Spring MVC 是一种基于Spring框架的Web开发框架,它是基于模型-视图-控制器(MVC)设计模式来组织和管理Web应用程序的开发的。数据库事务是保证数据操作的一致性和可靠性的重要机制。 在Spring MVC中,数据库事务的实现主要依赖于Spring框架内置的事务管理器。Spring框架为我们提供了多种事务管理器的实现方式,例如基于JDBC的DataSourceTransactionManager和基于JPA的JpaTransactionManager等。这些事务管理器实现了Spring的PlatformTransactionManager接口,它负责管理和控制事务的生命周期。 Spring MVC中使用注解@Transactional来标注需要参与事务管理的方法或类。被@Transactional标注的方法或类,会在运行时被Spring框架内置的AOP机制拦截,以实现事务的控制。当方法被调用时,Spring框架首先会检查当前线程是否已经有一个事务实例存在,如果不存在则创建一个新的事务,如果存在则加入已存在的事务中。 一旦事务创建成功,Spring框架会开始执行方法体中的业务逻辑操作。如果方法执行成功,事务将会继续提交,操作的结果将会永久保存到数据库中。如果方法执行失败或发生异常,Spring框架会回滚事务,将操作的结果恢复到之前的状态。这样可以确保在出现异常情况时,数据库的数据不会被污染或损坏。 事务的提交和回滚是由Spring框架的事务管理器来完成的。事务管理器负责管理事务的开始、提交和回滚等操作,并与底层的数据库连接进行交互。Spring框架还提供了各种配置选项,可以通过配置文件或注解来灵活地控制事务的传播行为、隔离级别、超时设置等。这些配置选项可以根据实际业务需求进行调整,以提供更好的性能和可靠性。 总而言之,Spring MVC中的数据库事务的实现原理是依赖于Spring框架提供的事务管理器和AOP机制。通过使用@Transactional注解,我们可以简单地将需要参与事务管理的方法或类进行标注,使其具备事务性的功能。这样可以实现对数据库操作的一致性和可靠性保证。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值