博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
开发多年,你知道Spring如何在一个事务中开启另一个事务吗?
阅读量:2198 次
发布时间:2019-05-02

本文共 1520 字,大约阅读时间需要 5 分钟。

上面提到的情景可能不常见,但是还是会有的,一旦遇到,如果业务比较复杂,就会很麻烦,但是还是有解决的方案的,比如将一个service方法拆成两个方法,也就是将两个操作的事务分开,但是这只适用与业务比较简单的,如果出现多次数据库的写操作,而我们调用的系统只需要其中一个写操作的最新数据,如果我们将它分开,那么如果调用目标系统出现异常的时候,那么之前的写操作就不能回滚了。

举个简单的例子:

@Servicepublic class ServiceA {	@Transactional	public void doSomething(){				向数据库中添加数据;				调用其他系统;	}}

这里就用伪代码来做示例了,当我们执行了“向数据库中添加数据”,我们去数据库中查询,发现并没有我们添加的数据,但是当我们的service这个方法执行完成之后,数据库中就有这条数据了,这是由于数据库的隔离性造成的。

spring中的事务注解@transactional提供了一个参数:

Propagation propagation() default Propagation.REQUIRED;

这个参数是定义spring事务的传递性的,默认值为required,也就是如果有事务,就加入事务,如果没有,就创建事务。这个参数的值有很多,例如REQUIRES_NEW,这个值就代表创建一个新的事务,与原来的事务分开。这个好像能解决我们的问题。

我们将刚刚那个方法修改一下:

@Servicepublic class ServiceA {	@Transactional	public void doSomething(){				insert();				调用其他系统;	}		@Transactional(propagation = Propagation.REQUIRES_NEW)	public void insert(){		向数据库中添加数据;	}}

执行之后,发现结果还是没有改变,必须要整体执行完成,数据库中数据才会出现,说明还是在一个事务中。

我们再将代码修改一下:

@Servicepublic class ServiceA {	@Autowired	private ServiceB serviceB;	@Transactional	public void doSomething(){				serviceB.insert();				调用其他系统;	}}
@Servicepublic class ServiceB {	@Transactional(propagation = Propagation.REQUIRES_NEW)	public void insert(){		向数据库中添加数据;	}}

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

当然spring其实也考虑这个,在spring的配置中,我们只需要添加标签 <aop:aspectj-autoproxy expose-proxy="true"/> 或者 <aop:config expose-proxy="true"> ,并且在代码的调用中要求使用代理对象去调用即可:

((ServiceA ) AopContext.currentProxy()).insert();

作者:Mazin

https://my.oschina.net/u/3441184/blog/893628

往期推荐:

一只 有深度 有灵魂 的公众号0.0

转载地址:http://otlub.baihongyu.com/

你可能感兴趣的文章
Java网络编程和NIO详解6:Linux epoll实现原理详解
查看>>
Java网络编程和NIO详解7:浅谈 Linux 中NIO Selector 的实现原理
查看>>
Java网络编程与NIO详解8:浅析mmap和Direct Buffer
查看>>
Java网络编程与NIO详解10:深度解读Tomcat中的NIO模型
查看>>
Java网络编程与NIO详解11:Tomcat中的Connector源码分析(NIO)
查看>>
深入理解JVM虚拟机1:JVM内存的结构与消失的永久代
查看>>
深入理解JVM虚拟机3:垃圾回收器详解
查看>>
深入理解JVM虚拟机4:Java class介绍与解析实践
查看>>
深入理解JVM虚拟机5:虚拟机字节码执行引擎
查看>>
深入理解JVM虚拟机6:深入理解JVM类加载机制
查看>>
深入了解JVM虚拟机8:Java的编译期优化与运行期优化
查看>>
深入理解JVM虚拟机9:JVM监控工具与诊断实践
查看>>
深入理解JVM虚拟机10:JVM常用参数以及调优实践
查看>>
深入理解JVM虚拟机11:Java内存异常原理与实践
查看>>
深入理解JVM虚拟机12:JVM性能管理神器VisualVM介绍与实战
查看>>
深入理解JVM虚拟机13:再谈四种引用及GC实践
查看>>
Spring源码剖析1:Spring概述
查看>>
Spring源码剖析2:初探Spring IOC核心流程
查看>>
Spring源码剖析3:Spring IOC容器的加载过程
查看>>
Spring源码剖析4:懒加载的单例Bean获取过程分析
查看>>