Spring中导致事务传播失效的情况(自调用、方法访问权限、异常处理不当、传播类型选择错误等。在实际开发中,务必确保事务方法正确配置)

文章目录

  • 1. 自调用(内部调用)导致事务失效
    • 示例:
    • 解决办法:
  • 2. 事务方法不是 `public` 修饰
    • 示例:
    • 解决办法:
  • 3. 未被 Spring 管理的对象
    • 示例:
    • 解决办法:
  • 4. 异常类型不匹配导致事务回滚失败
    • 示例:
    • 解决办法:
  • 5. 传播类型使用不当
    • 示例:
    • 解决办法:
  • 6. 数据库本身不支持事务
    • 解决办法:
  • 7. 使用了不支持事务的操作
    • 解决办法:
  • 8. 事务管理器未正确配置
    • 解决办法:
  • 9. 代理模式下的事务不生效
    • 解决办法:
  • 总结

在 Spring 中,事务传播机制可以有效管理事务的传播和行为。但在某些情况下,事务传播可能会失效,导致事务没有按预期进行传播和处理。以下是常见导致事务传播失效的几种情况:

1. 自调用(内部调用)导致事务失效

这是最常见的问题之一。当一个类的非事务方法内部调用自身的事务方法时,Spring 的事务代理无法起作用。这是因为 Spring 的事务管理是基于 AOP 实现的,它通过动态代理在方法执行前后插入事务逻辑。但如果是同一个类的内部调用,Spring 无法通过代理拦截这个调用,事务不会被启动。

示例:

public class MyService {

    @Transactional
    public void transactionalMethod() {
        // 事务逻辑
    }

    public void nonTransactionalMethod() {
        // 通过内部调用执行事务方法
        transactionalMethod();
    }
}

在上述代码中,如果在 nonTransactionalMethod() 中调用了 transactionalMethod()transactionalMethod() 中的事务不会生效,因为这是一个自调用,不经过 Spring 的代理机制。

解决办法:

将事务方法抽取到另一个类中,或者通过依赖注入的方式调用。

@Service
public class MyService {

    @Autowired
    private MyService self;

    public void nonTransactionalMethod() {
        // 使用 self 调用,事务才能生效
        self.transactionalMethod();
    }

    @Transactional
    public void transactionalMethod() {
        // 事务逻辑
    }
}

2. 事务方法不是 public 修饰

Spring 的事务管理是基于代理的,它只会对 public 方法进行代理。对于 privateprotected、或者 package-private 的方法,Spring 事务不会生效,因为代理类无法拦截对这些方法的调用。

示例:

@Transactional
private void transactionalMethod() {
    // 事务逻辑
}

在这种情况下,事务不会生效,因为 transactionalMethod() 不是 public 方法。

解决办法:

确保事务方法是 public 的。

@Transactional
public void transactionalMethod() {
    // 事务逻辑
}

3. 未被 Spring 管理的对象

只有被 Spring 容器管理的 Bean,事务代理才能起作用。如果方法是在一个非 Spring 管理的类中调用的,即使添加了 @Transactional 注解,事务也不会生效。

示例:

public class ExternalClass {
    @Transactional
    public void transactionalMethod() {
        // 事务逻辑
    }
}

如果 ExternalClass 不是由 Spring 容器托管,transactionalMethod() 上的事务将不起作用。

解决办法:

确保事务相关的 Bean 是通过 Spring 管理的,即确保它们是由 Spring 容器注入的,例如通过 @Service@Component 标注的类。

4. 异常类型不匹配导致事务回滚失败

Spring 的默认事务管理策略是只在遇到 unchecked 异常(RuntimeException)Error 时回滚事务。如果方法抛出了一个 checked 异常(如 IOException),事务不会回滚。

示例:

@Transactional
public void transactionalMethod() throws Exception {
    // 如果抛出 checked 异常,事务不会回滚
    throw new Exception("Checked Exception");
}

在上述示例中,方法抛出的是一个 Exception(checked 异常),默认情况下,Spring 不会回滚事务。

解决办法:

可以通过 rollbackFor 属性明确指定哪些异常需要触发事务回滚:

@Transactional(rollbackFor = Exception.class)
public void transactionalMethod() throws Exception {
    // 事务会回滚
    throw new Exception("Checked Exception");
}

5. 传播类型使用不当

事务传播类型决定了当一个事务方法调用另一个事务方法时,事务如何传播。如果传播类型配置不当,可能导致事务行为与预期不一致,甚至事务失效。

示例:

使用 Propagation.NOT_SUPPORTEDPropagation.REQUIRES_NEW 传播类型时,如果子事务方法被配置为不支持或需要新的事务,可能会导致当前事务的逻辑与原始事务隔离,无法实现事务传播效果。

@Transactional(propagation = Propagation.NOT_SUPPORTED)
public void nonTransactionalMethod() {
    // 该方法不会加入现有事务,事务逻辑不会生效
}

解决办法:

根据业务需求选择正确的传播类型,例如使用 Propagation.REQUIRED 让子方法加入当前事务。

6. 数据库本身不支持事务

并非所有数据库都支持事务操作。比如,某些 NoSQL 数据库(如 MongoDB、Cassandra)对事务的支持是有限的。如果数据库不支持事务,即使在代码中配置了事务管理,事务也不会生效。

解决办法:

确保所使用的数据库支持事务,并正确配置数据库连接。

7. 使用了不支持事务的操作

某些数据库操作或框架方法不支持事务。比如,批量操作或者直接使用 JdbcTemplate 进行大批量数据插入时,可能不会受到事务控制的影响。

解决办法:

确保使用支持事务管理的数据库操作工具,如 JPA、Hibernate,或者正确配置 JdbcTemplate

8. 事务管理器未正确配置

Spring 事务管理依赖于 PlatformTransactionManager,如果没有正确配置事务管理器,事务也不会生效。常见问题包括未正确配置数据源或没有配置事务管理器。

解决办法:

确保 PlatformTransactionManager 正确配置,并且事务管理器与数据源绑定一致。

@Bean
public PlatformTransactionManager transactionManager(DataSource dataSource) {
    return new DataSourceTransactionManager(dataSource);
}

9. 代理模式下的事务不生效

Spring 使用动态代理(JDK 动态代理或 CGLIB)来实现 AOP。如果使用了 JDK 动态代理,它只对接口的方法进行代理,导致如果在实现类上直接调用方法,事务可能不生效。

解决办法:

如果没有接口,确保 @Transactional 注解的类使用的是 CGLIB 代理,而不是 JDK 代理。

@EnableTransactionManagement(proxyTargetClass = true) // 强制使用 CGLIB 代理

总结

事务失效的原因主要包括自调用、方法访问权限、异常处理不当、传播类型选择错误等。在实际开发中,务必确保事务方法正确配置,并且在调用方法时遵循 Spring AOP 的代理机制,避免事务失效。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/897364.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

GEE引擎架设好之后进游戏时白屏的解决方法——gee引擎白屏修复

这两天测试GeeM2引擎的服务端,最常见的问题就是点击开始游戏出现白屏,最早还以为是服务端问题,结果是因为升级了引擎,而没有升级NewUI这份文件导致的。解决方法如下: 下载GEE引擎包最新版,(可以…

C++实现精简实用的json解析库

fcjson.h #pragma once#if defined(WIN32) || defined(_WIN32) || defined(__WIN32__) || defined(__NT__) #define _CRT_SECURE_NO_WARNINGS #endif#include <stdint.h> #include <string> #include <vector> #include <map>// VS 将执行字符集指定为…

Jupyter Notebook修改你的默认路径

Jupyter Notebook修改你的默认路径 看到网上一堆抄来抄去的博客&#xff0c;图片和文字都不对应&#xff0c;搞得配置得头昏脑涨的&#xff0c;所以写了这个。 1.打开 Anaconda Prompt 2.输入 jupyter notebook --generate-config&#xff0c;注意提示后输入N&#xff0c;因为…

【K8s】Kubernetes 证书管理工具 Cert-Manager

本文内容均来自个人笔记并重新梳理&#xff0c;如有错误欢迎指正&#xff01; 如果对您有帮助&#xff0c;烦请点赞、关注、转发、订阅专栏&#xff01; 专栏订阅入口 | 精选文章 | Kubernetes | Docker | Linux | 羊毛资源 | 工具推荐 | 往期精彩文章 【Docker】&#xff08;全…

校招基础知识详解——计算机操作系统(内存管理)

文章目录 虚拟内存分页系统地址映射页面置换算法最佳页面置换算法(OPT, Optimal replacement algorithm)先进先出置换算法&#xff08;FIFO, First In First Out&#xff09;最近最久未使用的置换算法&#xff08;LRU, Least Recently Used&#xff09;最不常用算法最近未使用&…

Excel常用操作培训

1 Excel基本操作 1.1 常用快捷键 1.1.1快捷键操作工作簿、工作表 1.1.2快捷键操作 1.1.3单元格操作 1.1.4输入操作 2.1 常见功能描述 2.1.1 窗口功能栏 excel有很多功能可以用&#xff0c;新建文档后&#xff0c;可以最上方&#xff0c;可以看到所有的功能栏目 2.1.2 剪切板…

《YOLO目标检测》—— YOLO v2 详细介绍

&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;未写完&#xff01;&#xff01;&#xff01;&#xff0…

Android中导入讯飞大模型ai智能系统

1.在讯飞大平台申请免费接口(申请后获取url和token) 2.创建一个数据库进行储存对话聊天记录 package com.example.myapplication.XL; import android.content.ContentValues; import android.content.Context; import android.database.Cursor; import android.database.sqlit…

【SQL】SQL函数

&#x1f4e2; 前言 函数 是指一段可以直接被另一段程序调用的程序或代码。主要包括了以下4中类型的函数。 字符串函数数值函数日期函数流程函数 &#x1f384; 字符串函数 ⭐ 常用函数 函数 功能 CONCAT(S1,S2,...Sn) 字符串拼接&#xff0c;将S1&#xff0c;S2&#xff0…

Mongodb基础用法【总结】

关系型数据库和非关系型数据库的区别 关系型数据库 1.在关系型数据库中&#xff0c;数据都是存储在表中的&#xff0c;对存储的内容有严格的要求 2.因为我们在创建表的时候久已经规定了表中的字段 存储的数据类型 是否为空 唯一标识等规则 3.由于操作的都是结构化的数据&#…

一款.NET开源的i茅台自动预约小助手

前言 今天大姚给大家分享一款.NET开源、基于WPF实现的i茅台APP接口自动化每日自动预约&#xff08;抢茅台&#xff09;小助手&#xff1a;HyggeImaotai。 项目介绍 该项目通过接口自动化模拟i茅台APP实现每日自动预约茅台酒的功能&#xff0c;软件会在指定时间开始对管理的用…

【UE5】将2D切片图渲染为体积纹理,最终实现使用RT实时绘制体积纹理【第六篇-阶段总结篇】

因为马上就要进入下一个阶段&#xff0c;制作动态编辑体积纹理的模块。 但在这之前&#xff0c;要在这一章做最后一些整理。 首先&#xff0c;我们完成没完成的部分。其次&#xff0c;最后整理一下图表。最后&#xff0c;本文附上正在用的贴图 完善Shader 还记得我们之前注…

HBuilder X 中Vue.js基础使用1(三)

一、 模板语法 Vue 使用一种基于 HTML 的模板语法&#xff0c;使我们能够声明式地将其组件实例的数据绑定到呈现的 DOM 上。所有的 Vue 模板都是语法层面合法的 HTML&#xff0c;可以被符合规范的浏览器和 HTML 解析器解析。 英文官网: Vue.js - The Progressive JavaScript Fr…

浪潮云启操作系统(InLinux)bcache缓存实践:理解OpenStack环境下虚拟机卷、Ceph OSD、bcache设备之间的映射关系

前言 在OpenStack平台上&#xff0c;采用bcache加速ceph分布式存储的方案被广泛用于企业和云环境。一方面&#xff0c;Ceph作为分布式存储系统&#xff0c;与虚拟机存储卷紧密结合&#xff0c;可以提供高可用和高性能的存储服务。另一方面&#xff0c;bcache作为混合存储方案&…

Turn-it:优化线材重构雕塑制造

&#x1f428;文章摘要abstract 电线雕塑在工业应用和日常生活中都很重要。 本文提出了一种新的制造策略&#xff0c;通过调整目标形状以适应电线弯曲机&#xff0c;然后由人工将其弯曲回目标形状。&#xff08;机器弯曲人工弯曲&#xff09; 该方法通过两阶段弯曲策略实现&a…

力扣——用队列实现栈(C语言)

目录 题目&#xff1a; 原理&#xff1a; 结构体MyStack 出栈void myStackPop(MyStack* obj) 入栈void myStackPush(MyStack* obj, int x) 读取栈顶元素int myStackTop(MyStack* obj) 判断栈空bool myStackEmpty(MyStack* obj) 销毁栈void myStackFree(MyStack* obj) 整…

NewStar CTF 2024 Week1,Week2部分

WP部分学习官方解题思路&#xff0c;这次比赛还是收获满满呀 web方向&#xff1a; headach3 抓包拿到flag 会赢吗 第一关&#xff1a; 查看源码看到flag第一部分和目录 第二关&#xff1a; 查看js文件 revealflag方法传入了一个className参数 <script>async func…

8.three.js相机详解

8.three.js相机详解 1、 认识相机 在Threejs中相机的表示是THREE.Camera&#xff0c;它是相机的抽象基类&#xff0c;其子类有两种相机&#xff0c;分别是正投影相机THREE.OrthographicCamera和透视投影相机THREE.PerspectiveCamera&#xff1a; 正投影和透视投影的区别是&am…

燕山大学23级经济管理学院 10.18 C语言作业

燕山大学23级经济管理学院 10.18 C语言作业 文章目录 燕山大学23级经济管理学院 10.18 C语言作业1C语言的基本数据类型主要包括以下几种&#xff1a;为什么设计数据类型&#xff1f;数据类型与知识体系的对应使用数据类型时需要考虑的因素 21. 逻辑运算符2. 真值表3. 硬件实现4…

最大公约数(公式法)

求多个数的最大公约数 采用连续求gcd的方式 题目 ACCODE #include<bits/stdc.h> using namespace std; long long num[4]; int main(){cin>>num[1]>>num[2]>>num[3];sort(num1,num4);// cout<<num[1]<<" "<<num[2]<&…