解决SpringBoot中使用@Transactional注解遇到的问题

目录
  • 使用@Transactional注解遇到的问题
    • 1、不建议在接口上添加@Transactional注解
    • 2、@Transactional注解
    • 3、默认情况下
    • 4、数据库引擎需要支持事务管理
    • 5、同一类中methodA()方法
  • springboot 注解transactional失效
    • 1、在方法中捕获了异常
    • 2、spring中事务是代理模式
    • 3、A方法如果有事务注解
    • 4、本类中A方法调用

使用@Transactional注解遇到的问题

1、不建议在接口上添加@Transactional注解

一般在service类标签上添加@Transactional即可

2、@Transactional注解

只能应用到public可见度的方法上。如果应用到protected、private或者package可见度的方法上时,不会报错,但事务也不会起作用

3、默认情况下

spring会对uncheck异常进行事务回滚的;如果是checked异常则不会回滚,可添加注解 @Transactional(rollbackFor=Exception.class) 是的checked异常回滚。

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

4、数据库引擎需要支持事务管理

如果是mysql,注意表要使用事务的引擎,比如innodb,如果是myisam,事务不会起作用的

5、同一类中methodA()方法

没有@Transactional 注解,在其内部调用有@Transactional 注解的方法,有@Transactional 注解的方法methodB()的事务被忽略,不会开启新的事务,也不会发生回滚。例如:

@Service
public class TransactionService {
public void methodA(){
    this.methodB();
}
@Transactional
public void methodB(){
}
}

原因:Spring采用动态代理(AOP)实现对bean的管理和切片,它为我们的每个class生成一个代理对象。只有在代理对象之间进行调用时,可以触发切面逻辑。在 Spring 的 AOP 代理下,只有目标方法由外部调用,目标方法才由 Spring 生成的代理对象来管理。

详解:Spring的事务管理是通过AOP实现的,其AOP的实现对于非final类是通过cglib这种方式,即生成当前类的一个子类作为代理类,然后在调用其下的方法时,会判断这个方法有没有@Transactional注解,如果有的话,则会开启一个新的事务,并通过动态代理实现事务管理(拦截方法调用,执行事务等切面)。

当methodA()中调用methodB()时,并不是使用的代理对象,而是普通的javabean,从而导致this.methodB()时也不是代码对象,从而导致@Transactional失败,即发现methodA()上并没有@Transactional注解,所以整个AOP代理过程(事务管理)不会发生。

解决办法:

1、把这两个方法分开到不同的类中;

2、把注解@Transactional加到类名上面去;

3、把注解@Transactional加到methodA()方法上,methodB()不添加注解,在调用methodB()时两个方法的事务都会生效,因为methodA()默认的事务传播属性为PROPAGATION_REQUIRED,此时methodB()会加入到methodA()中

@Service
public class TransactionService {
@Transactional
public void methodA(){
    this.methodB();
}
public void methodB(){
}
}

4、获取本对象的代理对象,再进行调用。具体操作如:

  • Spring-content.xml上下文中,增加配置:<aop:aspectj-autoproxy expose-proxy="true"/>
  • 在TransactionService 中,用(transactionService )(AopContext.currentProxy()),获取到TransactionService 的代理类,再调用事务方法,强行经过代理类,激活事务切面。

springboot 注解transactional失效

1、在方法中捕获了异常

那么该方法事务不会回滚。本类中A方法(有事务注解)调用其他类中B方法(有事务注解,该类在容器中),B方法中捕获了异常,事务失效不可以回滚,在B方法中的catch里进行异常抛出则可以正常回滚。

本类中A方法(有事务注解)调用其他类中B方法(有事务注解,该类在容器中),A方法中捕获了异常,spring事务机制检测到异常,进行回滚。

所以最好在业务层catch中抛出异常让控制层捕获。

2、spring中事务是代理模式

在该类中A方法调用本类中B方法,A方法上没有事务注解,B方法上有事务注解,则B方法中出现异常,B方法事务不会回滚,A方法没有事务注解也不会回滚。

3、A方法如果有事务注解

调用了本类B方法,B方法出现异常也整体可回滚。

4、本类中A方法调用

其他类中B方法(有事务注解,该类在容器中),B方法发生异常,可以回滚。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • 详解在SpringBoot中@Transactional事物操作和事物无效问题排查

    目录 1.spring事务管理简述 2.SpringBoot中使用@Transactional注解 2.1.开启事务注解 2.2.在目标类.方法上添加注解@Transactional 2.3.细化事务配置 3.@Transactional事务实现机制 3.1.整体事务控制流程 3.2.Spring AOP的两种代理 3.3.事务操作的底层实现 4.@Transactional使用注释实现及问题排查 4.1.数据库引擎是否支持事务? 4.3.注解所在的类是否被加载成Bean? 4.2.注解所在方法

  • spring @Transactional 无效的解决方案

    关于@Transactional注解 一般都认为要注意以下三点 1 .在需要事务管理的地方加@Transactional 注解.@Transactional 注解可以被应用于接口定义和接口方法.类定义和类的 public 方法上 . 2 . @Transactional 注解只能应用到 public 可见度的方法上 . 如果你在 protected.private 或者 package-visible 的方法上使用 @Transactional 注解,它也不会报错, 但是这个被注解的方法将不会展

  • springboot中使用@Transactional注解事物不生效的坑

    一:在springboot中使用事物遇到的坑 1.我们知道spring中的事物分为两种:一种是编程式事物,一种是声明式事物.顾名思义,编程式事物是指通过代码去实现事物管理,这里不做过多说明.另一种是声明式事物,分为两种情况01:一种是通过传统xml方式配置,02:使用@Transaction注解方式配置,这是主要讲解的是通过注解方式配置.因为在springboot项目中,会自动配置DataSourceTransactionManager,我们只需要在对应的方法上或者类上加上@Transactio

  • 解决SpringBoot中使用@Transactional注解遇到的问题

    目录 使用@Transactional注解遇到的问题 1.不建议在接口上添加@Transactional注解 2.@Transactional注解 3.默认情况下 4.数据库引擎需要支持事务管理 5.同一类中methodA()方法 springboot 注解transactional失效 1.在方法中捕获了异常 2.spring中事务是代理模式 3.A方法如果有事务注解 4.本类中A方法调用 使用@Transactional注解遇到的问题 1.不建议在接口上添加@Transactional注解

  • 解决SpringBoot中使用@Async注解失效的问题

    错误示例,同一个类中使用异步方法: package com.xqnode.learning.controller; import com.fasterxml.jackson.core.JsonProcessingException; import org.springframework.scheduling.annotation.Async; import org.springframework.web.bind.annotation.GetMapping; import org.springf

  • 解决springboot中配置过滤器以及可能出现的问题

    在springboot添加过滤器有两种方式: 1.通过创建FilterRegistrationBean的方式(建议使用此种方式,统一管理,且通过注解的方式若不是本地调试,如果在filter中需要增加cookie可能会存在写不进前端情况) 2.通过注解@WebFilter的方式 通过创建FilterRegistrationBean的方式创建多个filter以及设置执行顺序: 1.创建两个实现Filter接口的类TestFilter1 .TestFilter2 package com.aoxun.c

  • 解决SpringBoot中@Email报错问题

    JSR303校验相关 现象:在springboot中使用@Email注解进行数据校验时,报没有该注解的错误. 解决方法: 在pom.xml中加该配置 <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-validation</artifactId> </dependency> 测试成功 ==测试成功== ![i

  • SpringBoot中使用@Scheduled注解创建定时任务的实现

    在项目日常开发过程中,经常需要定时任务来帮我们做一些工作,如清理日志.定时任务的实现方法主要有 Timer.Quartz 以及 elastic-job Timer 实现定时任务 只执行一次的定时任务 Timer timer = new Timer(); timer.schedule(new TimerTask() { @Override public void run() { System.out.println("2000毫米后执行一次."); } }, 2000); timer.s

  • 详解SpringBoot中添加@ResponseBody注解会发生什么

    SpringBoot版本2.2.4.RELEASE. [1]SpringBoot接收到请求 ① springboot接收到一个请求返回json格式的列表,方法参数为JSONObject 格式,使用了注解@RequestBody 为什么这里要说明返回格式.方法参数.参数注解?因为方法参数与参数注解会影响你使用不同的参数解析器与后置处理器!通常使用WebDataBinder进行参数数据绑定结果也不同. 将要调用的目标方法如下: @ApiOperation(value="分页查询") @Re

  • 详解SpringBoot中@NotNull,@NotBlank注解使用

    目录 一.添加依赖 二.在类中使用验证注解 1.创建验证实体类(嵌套使用) 2.创建全局异常处理器,对message信息进行处理,并返回给前端 3.在controller中的使用 三.在方法参数中使用验证注解,与@RequsetParam注解同时使用,注意类上使用@Validated 四.自定义验证注解 一.添加依赖 <!-- spring-boot 2.3及以上的版本只需要引入下面的依赖 --> <dependency> <groupId>org.springfram

  • 解决springboot中@DynamicUpdate注解无效的问题

    springboot 中 @DynamicUpdate 注解无效解决方案 遇到的问题 项目中使用 jpa,以前没用过,所以踩坑在所难免. 在使用过程中,要更新一条记录的某个字段,更新成功以后,发现整条记录只剩下我更新的那个字段,其他的全部为空了. 瞬间明白,这种更新是全覆盖,针对每个字段 update,实体类没赋值的字段,也直接将空值 set 过去了. 寻求解决方案 正在庆幸这么容易就解决,突然发现并没有这么简单. 群众的力量是无穷大的,我立刻就明白这个注解为什么无效,原来是搞错了它的用途. 一

  • 解决springboot+activemq启动报注解错误的问题

    springboot+activemq启动报注解错误 Description: Field jmsMessagingTemplate in com.haozz.demo.mq.PromoteActProducer required a bean of type 'org.springframework.jms.core.JmsMessagingTemplate' that could not be found. The injection point has the following anno

随机推荐