前言
本文主要介绍下spring事务中的传播行为。事务传播行为是spring框架独有的事务增强特性,他不属于的事务实际提供方数据库行为。这是spring为我们提供的强大的工具箱,使用事务传播行可以为我们的开发工作提供许多便利。
下面话不多说了,来一起看看详细的介绍吧
事务传播行为介绍
spring中的7个事务传播行为:
|事务行为|说明 |
|:--|:--|
|propagation_required | 支持当前事务,假设当前没有事务。就新建一个事务 |
| propagation_supports |支持当前事务,假设当前没有事务,就以非事务方式运行 |
| propagation_mandatory| 支持当前事务,假设当前没有事务,就抛出异常|
| propagation_requires_new | 新建事务,假设当前存在事务。把当前事务挂起|
|propagation_not_supported | 以非事务方式运行操作。假设当前存在事务,就把当前事务挂起 |
| propagation_never | 以非事务方式运行,假设当前存在事务,则抛出异常 |
| propagation_nested |如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则执行与propagation_required类似的操作。 |
举例说明
案例代码
servicea
1
2
3
4
5
|
servicea { void methoda() { serviceb.methodb(); } } |
serviceb
1
2
3
4
|
serviceb { void methodb() { } } |
1.propagation_required
假如当前正要运行的事务不在另外一个事务里,那么就起一个新的事务 比方说,serviceb.methodb的事务级别定义propagation_required, 那么因为执行servicea.methoda的时候,servicea.methoda已经起了事务。这时调用serviceb.methodb,serviceb.methodb看到自己已经执行在servicea.methoda的事务内部。就不再起新的事务。而假如servicea.methoda执行的时候发现自己没有在事务中,他就会为自己分配一个事务。这样,在servicea.methoda或者在serviceb.methodb内的不论什么地方出现异常。事务都会被回滚。即使serviceb.methodb的事务已经被提交,可是servicea.methoda在接下来fail要回滚,serviceb.methodb也要回滚
2.propagation_supports
假设当前在事务中。即以事务的形式执行。假设当前不在一个事务中,那么就以非事务的形式执行
3propagation_mandatory
必须在一个事务中执行。也就是说,他仅仅能被一个父事务调用。否则,他就要抛出异常
4.propagation_requires_new
这个就比较绕口了。 比方我们设计servicea.methoda的事务级别为propagation_required,serviceb.methodb的事务级别为propagation_requires_new。那么当运行到serviceb.methodb的时候,servicea.methoda所在的事务就会挂起。serviceb.methodb会起一个新的事务。等待serviceb.methodb的事务完毕以后,他才继续运行。
他与propagation_required 的事务差别在于事务的回滚程度了。由于serviceb.methodb是新起一个事务,那么就是存在两个不同的事务。假设serviceb.methodb已经提交,那么servicea.methoda失败回滚。serviceb.methodb是不会回滚的。假设serviceb.methodb失败回滚,假设他抛出的异常被servicea.methoda捕获,servicea.methoda事务仍然可能提交。
5.propagation_not_supported
当前不支持事务。比方servicea.methoda的事务级别是propagation_required 。而serviceb.methodb的事务级别是propagation_not_supported ,那么当执行到serviceb.methodb时。servicea.methoda的事务挂起。而他以非事务的状态执行完,再继续servicea.methoda的事务。
6.propagation_never
不能在事务中执行。
如果servicea.methoda的事务级别是propagation_required。 而serviceb.methodb的事务级别是propagation_never ,那么serviceb.methodb就要抛出异常了。
7.propagation_nested
如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则执行与propagation_required类似的操作。
spring中事务的配置
配置文件的方式
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
<tx:advice id= "txadvice" transaction-manager= "txmanager" > <tx:attributes> <!--设置所有匹配的方法,然后设置传播级别和事务隔离--> <tx:method name= "save*" propagation= "required" /> <tx:method name= "add*" propagation= "required" /> <tx:method name= "create*" propagation= "required" /> <tx:method name= "insert*" propagation= "required" /> <tx:method name= "update*" propagation= "required" /> <tx:method name= "merge*" propagation= "required" /> <tx:method name= "del*" propagation= "required" /> <tx:method name= "remove*" propagation= "required" /> <tx:method name= "put*" propagation= "required" /> <tx:method name= "get*" propagation= "supports" read-only= "true" /> <tx:method name= "count*" propagation= "supports" read-only= "true" /> <tx:method name= "find*" propagation= "supports" read-only= "true" /> <tx:method name= "list*" propagation= "supports" read-only= "true" /> <tx:method name= "*" propagation= "supports" read-only= "true" /> </tx:attributes> </tx:advice> |
注解的方式
1
2
|
<!--开启注解的方式--> <tx:annotation-driven transaction-manager= "transactiomanager" /> |
@transactional(propagation=propagation.required)
如果有事务, 那么加入事务, 没有的话新建一个(默认情况下)
@transactional(propagation=propagation.not_supported)
容器不为这个方法开启事务
@transactional(propagation=propagation.requires_new)
不管是否存在事务,都创建一个新的事务,原来的挂起,新的执行完毕,继续执行老的事务
@transactional(propagation=propagation.mandatory)
必须在一个已有的事务中执行,否则抛出异常
@transactional(propagation=propagation.never)
必须在一个没有的事务中执行,否则抛出异常(与propagation.mandatory相反)
@transactional(propagation=propagation.supports)
如果其他bean调用这个方法,在其他bean中声明事务,那就用事务.如果其他bean没有声明事务,那就不用事务.
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对服务器之家的支持。
原文链接:https://www.cnblogs.com/dengpengbo/p/10425316.html