首页 > 其他分享 >Spring事务@Transaction失效原因

Spring事务@Transaction失效原因

时间:2023-12-26 18:25:18浏览次数:37  
标签:事务 Transaction Spring Transactional order 失效 updateOrder public

目录

1、数据库引擎不支持事务

某些数据库引擎不支持事务,如果你使用这些引擎,则不能正确地使用@Transactional注解。

2、事务管理器配置问题

如果 Spring 配置文件中没有启用事务注解配置,或者配置不正确,事务不会生效。

或者开启了多个事务管理器,而事务注解没有指定使用哪个事务管理器,事务注解可能无法生效。确保事务注解中指定了正确的事务管理器。

3、没有被 Spring IOC管理

使用@Transactional进行事务管理时,Spring需要通过代理对象来管理事务。如果你的类没有被代理,则事务注解将无法生效。

// @Service
public class OrderServiceImpl implements OrderService {

    @Transactional
    public void updateOrder(Order order) {
        // update order
    }
}

如果此时把 @Service 注解注释掉,这个类就不会被加载成一个 Bean,那这个类就不会被 Spring 管理了,事务自然就失效了。

4、非public方法

以下来自 Spring 官方文档:

When using proxies, you should apply the @Transactional annotation only to methods with public visibility. If you do annotate protected, private or package-visible methods with the @Transactional annotation, no error is raised, but the annotated method does not exhibit the configured transactional settings. Consider the use of AspectJ (see below) if you need to annotate non-public methods.

意思就是 @Transactional 只能用于 public 的方法上,否则事务不会失效,如果要用在非 public 方法上,可以开启 AspectJ 代理模式。

具体原因是因为Spring事务是基于AOP代理的,而代理模式需要方法是public才能进行代理。

5、内部方法调用(常见)

事务的生效是基于AOP代理的,如果在同一个类中的一个方法内调用同类中的另一个方法,事务注解可能不会起作用,因为代理机制不会被激活。

来看两个示例:

@Service
public class OrderServiceImpl implements OrderService {

    public void update(Order order) {
        updateOrder(order);
    }

    @Transactional
    public void updateOrder(Order order) {
        // update order
    }

}

update方法上面没有加 @Transactional 注解,调用有 @Transactional 注解的 updateOrder 方法,updateOrder 方法上的事务会生效吗?(答案:不生效)

再来看下面这个例子:

@Service
public class OrderServiceImpl implements OrderService {

    @Transactional
    public void update(Order order) {
        updateOrder(order);
    }

    @Transactional(propagation = Propagation.REQUIRES_NEW)
    public void updateOrder(Order order) {
        // update order
    }

}

这次在 update 方法上加了 @Transactional,updateOrder 加了 REQUIRES_NEW 新开启一个事务,那么新开的事务管用么?

这两个例子的答案是:不管用!

解决方法:

1、通过将该方法移动到一个单独的类中,并通过依赖注入调用来解决这个问题。

2、在的类中注入自己,用注入的对象再调用另外一个方法。

6、异常被捕获(常见)

如果在事务方法中捕获了异常并处理了它,但没有将其重新抛出,那么事务将不会回滚。

例子:

@Service
public class OrderServiceImpl implements OrderService {

    @Transactional
    public void updateOrder(Order order) {
        try {
            // update order
        } catch {

        }
    }
}

7、非受检异常(常见)

Spring默认只会在遇到非受检异常(继承自RuntimeException)时回滚事务。如果你的事务方法抛出了受检异常(继承自Exception),则事务可能不会回滚。

可以通过在@Transactional注解中指定rollbackFor属性来指定回滚的异常类型。

@Service
public class OrderServiceImpl implements OrderService {

    @Transactional
    public void updateOrder(Order order) {
        try {
            // update order
        } catch {
            throw new Exception("更新错误");
        }
    }
}

这样事务也是不生效的,因为默认回滚的是:RuntimeException

如果你想触发其他异常的回滚,需要在注解上配置一下,如:

@Transactional(rollbackFor = Exception.class)
 public void updateOrder(Order order) {
   try {
     // update order
   } catch {
     throw new Exception("更新错误");
   }
 }

这些是一些可能导致Spring事务@Transactional不生效的常见原因。但要注意,每个问题都需要具体分析和调试,以确定根本原因。如果问题仍然存在,请仔细检查你的代码、配置和环境,并查阅相关文档和资料来解决问题。

标签:事务,Transaction,Spring,Transactional,order,失效,updateOrder,public
From: https://www.cnblogs.com/luler/p/17928996.html

相关文章

  • Spring应用最常用的工具类汇总
    文件资源访问假设有一个文件地址位于Web应用的类路径下,您可以通过以下方式对这个文件资源进行访问:FileSystemResource以文件系统绝对路径的方式进行访问;ClassPathResource以类路径的方式进行访问;ServletContextResource以相对于Web应用根目录的方式进行访问。Resourc......
  • 浅聊springboot的启动流程
    前言本文会对Springboot启动流程进行详细分析。但是请注意,Springboot启动流程是Springboot的逻辑,请千万不要将Springboot启动流程相关逻辑与Spring的相关逻辑混在一起,比如把Spring的bean生命周期的逻辑混在Springboot启动流程中,那么整个体系就复杂且混乱了。所以本文仅重点关注Spri......
  • SpringBoot实现热部署详解
    前言SpringBoot热部署是一种开发时极为有用的功能,它能够让开发人员在代码修改后无需手动重启应用程序就能立即看到变化的效果。以下是使用SpringBoot热部署的几个主要原因:1.提高开发效率热部署使开发人员能够更快地验证和测试他们的代码更改。无需手动重启应用程序,每次修改后只需......
  • springboot自动部署脚本
    #!/bin/sh##javaenvexportJAVA_HOME=/usr/local/jdk/jdk1.8.0_101exportJRE_HOME=$JAVA_HOME/jreAPI_NAME=apiJAR_NAME=$API_NAME\.jar#PID代表是PID文件PID=$API_NAME\.pid#使用说明,用来提示输入参数usage(){echo"Usage:sh执行脚本.sh[start|stop|re......
  • SpringBoot集成多个RabbitMq(多个MQ链接)
    ##2023年12月16日20:25:36 项目中使用RabbitMQ作为应用间信息互通,本次梳理下关于MQ的使用。1、引入依赖<!--引入依赖,使用v2.5.6版本--><dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot......
  • 基于SpringBoot+Vue的毕业设计系统的开发设计实现(源码+lw+部署文档+讲解等)
    (文章目录)前言:heartpulse:博主介绍:✌全网粉丝10W+,CSDN特邀作者、博客专家、CSDN新星计划导师、全栈领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java、小程序技术领域和毕业项目实战✌:heartpulse:......
  • Spring MVC 源码分析 - 一个请求的旅行过程
    在上一篇《WebApplicationContext容器的初始化》文档中分析了SpringMVC是如何创建两个容器的,其中创建RootWebApplicationContext 后,调用其refresh()方法会触发刷新事件,完成SpringIOC初始化相关工作,会初始化各种SpringBean到当前容器中,该系列文档暂不分析我们先来了解一......
  • Spring学习记录之Spring对IoC的实现
    Spring学习记录之Spring对IoC的实现前言这篇文章是我第二次学习b站老杜的spring相关课程所进行的学习记录,算是对课程内容及笔记的二次整理,以自己的理解方式进行二次记录,其中理解可能存在错误,欢迎且接受各位大佬们的批评指正;关于本笔记,只是我对于相关知识遗忘时快速查阅了解使......
  • 浅聊SpringCloud的网关
    为什么要设计网关?上网搜罗了一下,觉得别人说的挺好,就引用了一下,在使用微服务的时候,不同的功能业务会集成一个服务群,而网关是基于服务群上的一个服务层,也是单独暴露给客户端的APIs。客户端对微服务的依赖直接使重构服务变得困难。一种直观的方法是将这些服务隐藏在一个新的服务层后面......
  • Spring Event 与 AOP
    在构建现代化的应用中,日志记录是不可或缺的一环。Spring框架为我们提供了强大的事件机制(SpringEvent)和切面编程(AOP),结合使用可以实现优雅的日志记录,使得代码更加模块化和可维护。本文将介绍如何结合SpringEvent和AOP,以及如何在不同场景下应用这两个强大的特性。1.SpringEvent......