IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> Spring事务失效场景 -> 正文阅读

[Java知识库]Spring事务失效场景

Spring事务失效场景

template

前言

今天遇到事务失效的问题,找了下事务失效的问题 ,发现事务失效的情况还是挺多的,有时候不注意就会导致问题,本篇文章进行整理总结了常见的事务失效的情况。

博客说明

文章所涉及的资料来自互联网整理和个人总结,意在于个人学习和经验汇总,如有什么地方侵权,请联系本人删除,谢谢!

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

这里以 MySQL 为例,其 MyISAM 引擎是不支持事务操作的,InnoDB 才是支持事务的引擎,一般要支持事务都会使用 InnoDB。

根据 MySQL 的官方文档:

https://dev.mysql.com/doc/refman/5.5/en/storage-engine-setting.html

MySQL 5.5.5 开始的默认存储引擎是:InnoDB,之前默认的都是:MyISAM,所以这点要值得注意,底层引擎不支持事务再怎么搞都是白搭。

2.没有被 Spring 管理

如下所示:

// @Service  
public class DeptTransactionServiceImpl implements DeptService {

	@Transactional
	public void updateDept(OrmDeptCondition deptCondition) {
		// update dept
	}
}

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

3.方法不是 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 代理模式。

4.自身调用问题

示例一如下

//示例 1
@Service
public class DeptTransactionServiceImpl implements DeptService {


	public void updateDept(OrmDeptCondition deptCondition) {
		updateDeptTest(deptCondition);
	}

	@Transactional
	public void updateDeptTest(OrmDeptCondition deptCondition) {
		// update dept
	}
}

示例二如下

@Service
public class DeptTransactionServiceImpl implements DeptService {

	@Transactional
	public void updateDept(OrmDeptCondition deptCondition) {
		updateDeptTest(deptCondition)
	}

	@Transactional(propagation = Propagation.REQUIRES_NEW)
	public void updateDeptTest(OrmDeptCondition deptCondition) {
		// update dept
	}
}

  • 示例1 中,updateDept方法上面没有加 @Transactional 注解,调用有 @Transactional 注解的 updateOrder 方法
  • 示例2 中,updateDept方法和updateDeptTest方法都加上注解

上面2种情况事务都会失效

因为他们自身调用,没有经过Spring的代理类,默认只有在外部调用事务才会生效。如何解决这种情况呢?请小伙伴们往下继续看

5.@Transactional的扩展配置不支持事务

Propagation.NOT_SUPPORTED:表示不以事务运行,当前若存在事务则挂起。这表示不支持以事务的方式运行,所以即使事务生效也是白搭!

@Service
public class DeptTransactionServiceImpl implements DeptService {

	@Transactional
	public void updateDept(OrmDeptCondition deptCondition) {
		updateDeptTest(deptCondition);
	}

	@Transactional(propagation = Propagation.NOT_SUPPORTED)
	public void updateDeptTest(OrmDeptCondition deptCondition) {
		// update dept
	}
}

扩展 Spring事务传播属性

在 Spring的 TransactionDefinition接口中一共定义了六种事务传播属性:

REQUIRED(0),支持当前事务,如果当前没有事务,就新建一个事务。这是最常见的选择。
SUPPORTS(1),支持当前事务,如果当前没有事务,就以非事务方式执行。 
MANDATORY(2),支持当前事务,如果当前没有事务,就抛出异常。 
REQUIRES_NEW(3),新建事务,如果当前存在事务,把当前事务挂起。 
NOT_SUPPORTED(4),以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。 
NEVER(5),非事务方式执行,如果当前存在事务,则抛出异常。 
NESTED(6);如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则进行与PROPAGATION_REQUIRED类似的操作

在第四步骤中:自身调用事务失效问题如何解决

@Service
public class ServiceA {

    @Autowired
    private ServiceB serviceB;
    @Transactional
    public void doSomething(){
        serviceB.insert();
        调用其他系统;
    }
}
@Service
public class ServiceB {

    @Transactional(propagation = Propagation.REQUIRES_NEW)
    public void insert(){
        向数据库中添加数据;
    }
}

我们将要事务分离出来的方法写在另一个service中,再次测试,发现执行完插入语句之后,数据库中就已经能查到数据了,说明事务分离了.

6.异常被吃了

把异常吃了,然后又不抛出来,事务也不会回滚!

@Service
public class DeptTransactionServiceImpl implements DeptService {

	@Transactional
	public void updateDept(OrmDeptCondition deptCondition) {
		try {
			updateDeptTest(deptCondition);
		} catch(Exception e) {

		}
	}
}

7.异常类型错误

@Service
public class DeptTransactionServiceImpl implements DeptService {

	@Transactional
   //@Transactional(rollbackFor = Exception.class)  
	public void updateDept(OrmDeptCondition deptCondition) throws Exception{
		try {
			updateDeptTest(deptCondition);
		} catch(Exception e) {
			throw new Exception("11");
		}
	}
}

这样事务也是不生效的,因为默认回滚的是:RuntimeException,如果你想触发其他异常的回滚,需要在注解上配置一下,如:

@Transactional(rollbackFor = Exception.class)

参考

Spring事务失效:https://mp.weixin.qq.com/s/Yi0HK1DbAxv3N3Y-Drmqtw

  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2022-03-31 23:49:43  更:2022-03-31 23:51:20 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/24 8:05:13-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码