Spring事务不生效问题
事务未生效可能造成严重的数据不一致性问题,因而保证事务生效至关重要。Spring事务是通过Spring aop实现的,所以不生效的本质问题是spring aop没生效,或者说没有代理成功,所以有必要了解下spring aop。
spring事务不生效场景
- 非public修饰方法
- private修饰
@Transactional
private void save(){...}
调用save方法,事务不生效 - protected修饰
@Transactional
protected void save(){...}
调用save方法,事务不生效
- private修饰
- final修饰的public法或类
- final修饰方法
@Transactional
public final void save(){...}
调用save方法,事务不生效 - final修饰类
@Transactional
public final class UserService{...}
调用UserService的方法,事务不生效
- final修饰方法
- 被代理对象内部方法调用
- 非事务方法调用内部事务方法
public void save(){
insert()
}
@Transactional
public void insert(){...}
此时调用save()方法,insert()上的事务不生效 - 事务方法调用内部事务方法
@Transactional
public void save(){
insert()
}
@Transactional
public void insert(){...}
此时调用save()方法,insert()上的事务不生效,save()方法正常调用时(非内部方法调用)事务生效
- 非事务方法调用内部事务方法
Spring aop的简单介绍
spring aop支持动态代理(运行时代理)和静态代理(编译期织入)
基本概念
- target对象
目标对象,即想要代理的目标。 - proxy
代理对象,访问目标对象的方法,需要通过代理对象,无法直接访问目标对象。
代理对象可能允许,最终调用目标对象方法,也可能不允许。
- target对象
静态代理(Aspectj)
1.该实现需要使用特殊编译器,一般不使用,spring只是提供相应的整合实现。
2.使用动态代理已经可以完成80%以上的需求了。
3.本文也是基于动态代理的情况下讨论事务不生效情况的。
4.因此在此不展开详细讨论。动态代理(CGLIB&JDK)
- JDK动态代理
1.JDK动态代理通过实现相同接口实现代理(Proxy)
2.spring管理的Bean中是对应的是Proxy
3.Proxy对象中包含target对象
4.目标方法可以执行时,总是调用target的原生方法
5.事务aop默认publicMethodsOnly(该配置未提供直接修改-AnnotationTransactionAttributeSource) - CGLIB动态代理
1.CGLIB动态代理通过继承目标类生成子类作为代理
2.spring管理的Bean中是对应的是Proxy
3.Proxy对象中包含target对象
4.目标方法可以执行时,总是调用target的原生方法
5.事务aop默认publicMethodsOnly(该配置未提供直接修改-AnnotationTransactionAttributeSource)
- JDK动态代理
Spring 事务不生效情况(即aop不生效情况),原因解析
- 无法代理
- 方法使用非public修饰(protected、private等等)
@Transactional
private void save(){...}
或
@Transactional
protected void save(){...}
spring 事务aop默认publicMethodsOnly,即只有public修饰的方法,会被代理,private和protected修饰的方法save()无法被代理,事务无效。 - 方法使用final修饰
@Transactional
public final void save(){...}
CGLIB动态代理通过生成子类的方式代理,final方法无法重写 - 类使用final修饰
@Transactional
public final class UserService{...}
CGLIB动态代理通过生成子类的方式代理,final类无法生成子类
- 方法使用非public修饰(protected、private等等)
- 未通过proxy对象调用
- 非事务方法调用内部事务方法
public void save(){
insert()
}
@Transactional
public void insert(){...}
该情况隐含使用this关键字,因此被代理对象的内部事务方法insert()直接被调用,而不经过代理对象;
内部方法insert()上的事务无效。 - 事务方法调用内部事务方法
@Transactional
public void save(){
insert()
}
@Transactional
public void insert(){...}
该情况隐含使用this关键字,因此被代理对象的内部事务方法insert()直接被调用,而不经过代理对象;
内部方法insert()上的事务无效,但是事务方法save()正确调用时(通过代理对象调用),save()方法的事务生效。
- 非事务方法调用内部事务方法
调试辅助
当无法通过代码分析确定是否有事务(aop是否代理成功)时,可以在开发环境调试确认
- 事务AOP
在配置了expose-proxy=true时,调用静态方法TransactionAspectSupport.currentTransactionStatus()可以获取当前事务信息;
该信息存储在ThreadLocal类型的线程变量;不存在时会抛异常,所以使用时需要try-catch。 - 其他AOP
在配置了expose-proxy=true时,调用静态方法AopContext.currentProxy()可以获取当前代理对象;该对象存储在ThreadLocal类型的线程变量;不存在时会抛异常,所以使用时需要try-catch。
Spring事务不生效问题的更多相关文章
- Spring基础系列-Spring事务不生效的问题与循环依赖问题
原创作品,可以转载,但是请标注出处地址:https://www.cnblogs.com/V1haoge/p/9476550.html 一.提出问题 不知道你是否遇到过这样的情况,在ssm框架中开发we ...
- mysql测试spring事务是否生效
同时对三张表进行插入操作,事务保证完整性.下面进行简单测试: 1. 锁定表 锁定用户表 LOCK TABLES user WRITE; 查看表是否锁定: show ; 显示被锁定的表. 2. 验证在同 ...
- [转]Spring事务嵌套引发的血案---Transaction rolled back because it has been marked as rollback-only
原文地址:https://blog.csdn.net/f641385712/article/details/80445912 1.概述 想必大家一想到事务,就想到ACID,或者也会想到CAP.但笔者今 ...
- spring中注解式事务不生效的问题
常用的解决方法可以百度,我针对我的问题描述一下 Mysql中InnoDB引擎才支持事务, MyISAM不支持事务. 当你尝试了各种方法解决spring中注解式事务不生效时, 一定要查看一下数据库中表的 ...
- 分析spring事务@Transactional注解在同一个类中的方法之间调用不生效的原因及解决方案
问题: 在Spring管理的项目中,方法A使用了Transactional注解,试图实现事务性.但当同一个class中的方法B调用方法A时,会发现方法A中的异常不再导致回滚,也即事务失效了. 当这个方 ...
- Spring中同一个service中方法相互调用事务不生效问题解决方案
问题描述: 我们在用Spring框架开发Web项目过程中,经常需要用同一个service中的一个方法调用另一个方法,如果此时调用方没有添加事务注解@Transactional,而在被调用方添加事务注解 ...
- 两个与spring事务相关的问题
有些spring相关的知识点之前一直没有仔细研究:比如spring的事务,并不是没有使用,也曾经简单的在某些需要事务处理的方法上通过增加事务注解来实现事务功能,仅仅是跟随使用(甚至并未测试过事务的正确 ...
- 【转】Spring事务管理
原文链接 在 Spring 中,事务是通过 TransactionDefinition 接口来定义的.该接口包含与事务属性有关的方法.具体如清单 1 所示: 清单 1. TransactionDefi ...
- Spring事务讲解示例
Spring 事务Transaction1.事务的属性1.1 事务隔离IsolationLevel1.2 事务传播PropagationBehavior1.3 事务超时Timeout1.4 只读状态R ...
随机推荐
- url传参和解决中文乱码
在A页面把参数传给B页面 index.html?name="张三" 在B页面接收(js) function getQueryString(name) { var result = ...
- .NET 6 中的HTTP 3支持
dotnet团队官方博客发布了一篇HTTP3的文章:HTTP/3 support in .NET 6:https://devblogs.microsoft.com/dotnet/http-3-supp ...
- vscode快速添加引号 批量增加引号(用于批量格式化代码)
一.在浏览器中将Params复制到pycharm的py文件中 二.选中需要添加引号的部分,Ctrl+H 调出替换工具栏 三.填写正则表达式 (.*?): (.*) '$1':'$2', 右侧注意点击使 ...
- R和Rstudio的安装
首先是安装R再安装Rstudio 链接放在这里: R语言软件以及Rstudio软件下载:链接:https://pan.baidu.com/s/11TH4mJjoi3QXGfamB697rw 密码:o1 ...
- jmeter之聚合报告(Aggregate Report)
jmeter最常用的listener--聚合报告Aggregate Report,每一个字段的具体含义是什么? Label:每个请求的名称.每个 JMeter 的 element(例如 HTTP Re ...
- php 设计模式 --适配器
1,目标:实现一个不同的类不同方法,符合一定的规范: 规范类 <?php interface Iplay{ function Attack(); function Defence(); } cl ...
- P4091-[HEOI2016/TJOI2016]求和【斯特林数,NTT】
正题 题目链接:https://www.luogu.com.cn/problem/P4091 题目大意 给出\(n\),求 \[\sum_{i=0}^n\sum_{j=0}^i\begin{Bmatr ...
- P5048-[Ynoi2019 模拟赛]Yuno loves sqrt technology III【分块】
正题 题目链接:https://www.luogu.com.cn/problem/P5048 题目大意 就是这个 [QA]区间众数,但空间很小 长度为\(n\)的序列,要求支持查找区间众数出现次数. ...
- NOI.AC#2144-子串【SAM,倍增】
正题 题目链接:http://noi.ac/problem/2144 题目大意 给出一个字符串\(s\)和一个序列\(a\).将字符串\(s\)的所有本质不同子串降序排序后,求有多少个区间\([l,r ...
- 二、mybatis之数据输出
上一篇我们做了一个入门案例,是我们做mybatis的基本步骤,不熟悉的可以回顾一下https://www.cnblogs.com/jasmine-e/p/15330355.html,在这篇文章中只是简 ...