这样的情景可能不常见,但是还是会有的,一旦遇到,如果业务比较复杂,就会很麻烦,但是还是有解决的方案的,比如将一个service方法拆成两个方法,也就是将两个操作的事务分开。
但是这只适用与业务比较简单的,如果出现多次数据库的写操作,而我们调用的系统只需要其中一个写操作的最新数据,如果我们将它分开,那么如果调用目标系统出现异常的时候,那么之前的写操作就不能回滚了。
举个简单的例子:南昌*【网址:bztss.com】
@Service
public class ServiceA {
@Transactional
public void doSomething(){
向数据库中添加数据;
调用其他系统;
}
}
这里就用伪代码来做示例了,当我们执行了“向数据库中添加数据”,我们去数据库中查询,发现并没有我们添加的数据,但是当我们的service这个方法执行完成之后,数据库中就有这条数据了,这是由于数据库的隔离性造成的。
Spring中的事务注解 @transactional 提供了一个参数:*公司【网址:bztss.com】
Propagation propagation() default Propagation.REQUIRED;
这个参数是定义 Spring事务的传递性的,默认值为:required,也就是如果有事务,就加入事务,如果没有,就创建事务。
这个参数的值有很多,例如:REQUIRES_NEW,这个值就代表创建一个新的事务,与原来的事务分开。这个好像能解决我们的问题。关注Java技术栈公众号在后台回复:spring,可获取一份栈长整理的最新 Spring 系列技术干货。
我们将刚刚那个方法修改一下:
@Service
public class ServiceA {
@Transactional
public void doSomething(){
insert();
调用其他系统;
}
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void insert(){
向数据库中添加数据;
}
}
执行之后,发现结果还是没有改变,必须要整体执行完成,数据库中数据才会出现,说明还是在一个事务中。Spring的核心思想,推荐你看看。
我们再将代码修改一下:阿克苏新闻
@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中,再次测试,发现执行完插入语句之后,数据库中就已经能查到数据了,说明事务分离了,完成了我们的需求。
当然 [Spring](http://mp.weixin.qq.com/s?天津*【网址:bztss.com】__biz=MzI3ODcxMzQzMw==&mid=2247491561&idx=2&sn=446e9a25108a71bbb8a0c24a955f3ada&chksm=eb539adfdc2413c9e80a4e0811b83b8978f0d2fc367e0ba60c70e36cce2a5c502c4d98623d95&scene=21#wechat_redirect)其实也考虑这个,在 [Spring](http://mp.weixin.qq.com/s?__biz=MzI3ODcxMzQzMw==&mid=2247491561&idx=2&sn=446e9a25108a71bbb8a0c24a955f3ada&chksm=eb539adfdc2413c9e80a4e0811b83b8978f0d2fc367e0ba60c70e36cce2a5c502c4d98623d95&scene=21#wechat_redirect)的配置中,我们只需要添加标签:
aop:aspectj-autoproxy expose-proxy="true"/
或者:
aop:config expose-proxy="true"
并且在代码的调用中要求使用代理对象去调用即可:
((ServiceA ) AopContext.currentProxy()).insert();
相关文章
- 04-10Spring 如何在一个事务中开启另一个事务?
- 04-10spring 事务处理中,同一个类中:A方法(无事务)调B方法(有事务),事务不生效问题
- 04-10spring 默认情况下事务是惟一的 同一个方法里面第一个sql开启后 在执行完 将事务传递给下一个sql
- 04-10使用spring声明式事务,spring使用AOP来支持声明式事务,会根据事务属性,自动在方法调用之前决定是否开启一个事务,并在方法执行之后决定事务提交或回滚事务。
- 04-10C# 最基本的涉及模式(单例模式) C#种死锁:事务(进程 ID 112)与另一个进程被死锁在 锁 | 通信缓冲区 资源上,并且已被选作死锁牺牲品。请重新运行该事务,解决方案: C#关闭应用程序时如何关闭子线程 C#中 ThreadStart和ParameterizedThreadStart区别
- 04-10分析spring事务@Transactional注解在同一个类中的方法之间调用不生效的原因及解决方案