spring的事务管理是类级别的,而不是方法级别的!
不管在spring配置文件中如何设置spring的传播propagation级别,同一个类的不同方法都在同一个事务中!!
代码说明一:
代码一:
@Service
public class TestServiceImpl implements TestService {
@Resource
private TestDAO testDAO;
/* (non-Javadoc)
* @see com.cums.pds.service.TestService#test3()
*/
@Override
public void test3() {
String SQL = "{ call TEST2 }";
testDAO.bulkUpdateBySQL(SQL);
testDAO.flush();
}
/* (non-Javadoc)
* @see com.cums.pds.service.TestService#test4()
*/
@Override
public void test4() {
test3();
String sql = "insert into tbl_test(merchant_no) values(6)";
testDAO.bulkUpdateBySQL(sql);
testDAO.flush();
throw new AppException("error..................");
}
public static void main(String[] args) {
ApplicationContext ctx = new ClassPathXmlApplicationContext("spring/applicationContext*.xml");
TestService s = (TestService) ctx.getBean("testServiceImpl");
s.test4();
}
}
在上面的代码中,即使将propagation设置为"REQUIRES_NEW",test4方法中的回滚也引起了test3方法的回滚。
代码说明二:
代码二:
@Service
public class TestService2Impl implements TestService2 {
// @Resource
// private TestDAO testDAO;
@Resource
private TestDAO2 testDAO2;
/* (non-Javadoc)
* @see com.cums.pds.service.TestService#test3()
*/
@Override
public void test3() {
String SQL = "{ call TEST2 }";
// testDAO.bulkUpdateBySQL(SQL);
// testDAO.flush();
testDAO2.bulkUpdateBySQL(SQL);
testDAO2.flush();
}
}
代码三:
@Service
public class TestServiceImpl implements TestService {
@Resource
private TestDAO testDAO;
@Resource
private TestService2 testService2;
/* (non-Javadoc)
* @see com.cums.pds.service.TestService#test4()
*/
@Override
public void test4() {
testService2.test3();
String sql = "insert into tbl_test(merchant_no) values(6)";
testDAO.bulkUpdateBySQL(sql);
testDAO.flush();
throw new AppException("error..................");
}
public static void main(String[] args) {
ApplicationContext ctx = new ClassPathXmlApplicationContext("spring/applicationContext*.xml");
TestService s = (TestService) ctx.getBean("testServiceImpl");
s.test4();
}
}
这种情况下,如果propagation设置为"REQUIRES_NEW",testService的test4方法回滚,testService2的test3方法并不回滚。
如果propagation设置为"NESTED",testService的test4方法回滚,testService2的test3方法也回滚。
要注意的是,是否使用同一个DAO类并不影响结果的。
把代码二中的testDAO2全部换成testDAO,不影响结果。
代码说明三:
代码四:
@Service
public class TestService2Impl implements TestService2 {
// @Resource
// private TestDAO testDAO;
@Resource
private TestDAO2 testDAO2;
/* (non-Javadoc)
* @see com.cums.pds.service.TestService#test3()
*/
@Override
public void test3() {
String SQL = "{ call TEST2 }";
// testDAO.bulkUpdateBySQL(SQL);
// testDAO.flush();
testDAO2.bulkUpdateBySQL(SQL);
testDAO2.flush();
throw new AppException("exception......");
}
}
代码五:
@Service
public class TestServiceImpl implements TestService {
@Resource
private TestDAO testDAO;
@Resource
private TestService2 testService2;
/* (non-Javadoc)
* @see com.cums.pds.service.TestService#test4()
*/
@Override
public void test4() {
String sql = "insert into tbl_test(merchant_no) values(6)";
testDAO.bulkUpdateBySQL(sql);
testDAO.flush();
testService2.test3();
// throw new AppException("error..................");
}
public static void main(String[] args) {
ApplicationContext ctx = new ClassPathXmlApplicationContext("spring/applicationContext*.xml");
TestService s = (TestService) ctx.getBean("testServiceImpl");
s.test4();
}
}
这种情况下,如果propagation设置为"REQUIRES_NEW",testService2的test3方法回滚,testService的test4方法也回滚。因为,testService2的test3方法抛出异常,testService的test4方法并没有处理这个异常,因此,testService的test4方法也产生了异常。因此都回滚。此时,如果对testService2的test3方法使用try。。。catch。。。,则testService的test4方法正常提交,testService2的test3方法回滚。
如果propagation设置为"NESTED",testService2的test3方法回滚,testService的test4方法也回滚。
REQUIRES_NEW和NESTED的区别在于:
外层事务异常回滚时,REQUIRES_NEW内层事务不回滚,因为它与外层事务相对独立;
NESTED内层事务回滚,因为它是外层事务的一部分。
但是,如果内层事务回滚,则外层事务如果不捕捉异常都会回滚,捕捉异常后外层事务不回滚。
感谢这篇文章:
spring事务传播级别 写道
1: PROPAGATION_REQUIRED
加入当前正要执行的事务不在另外一个事务里,那么就起一个新的事务
比如说,ServiceB.methodB的事务级别定义为PROPAGATION_REQUIRED, 那么由于执行ServiceA.methodA的时候,ServiceA.methodA已经起了事务,这时调用ServiceB.methodB,ServiceB.methodB看到自己已经运行在ServiceA.methodA的事务内部,就不再起新的事务。而假如ServiceA.methodA运行的时候发现自己没有在事务中,它就会为自己分配一个事务。这样,在ServiceA.methodA或者在ServiceB.methodB内的任何地方出现异常,事务都会被回滚。
2: PROPAGATION_SUPPORTS
如果当前在事务中,即以事务的形式运行,如果当前不在一个事务中,那么就以非事务的形式运行
3: PROPAGATION_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
理解Nested的关键是savepoint。它与PROPAGATION_REQUIRES_NEW的区别是,PROPAGATION_REQUIRES_NEW另起一个事务,将会与它的父事务相互独立,而Nested的事务和它的父事务是相依的,它的提交是要等和它的父事务一块提交的。也就是说,如果父事务最后回滚,它也要回滚的。
而Nested事务的好处是它有一个savepoint。
class ServiceA {
/*
* 事务属性配置为 PROPAGATION_REQUIRED
*/
void methodA() {
try {
// savepoint
ServiceB.methodB(); //PROPAGATION_NESTED 级别
} catch (Exception e) {
// 执行其它业务, 如 ServiceC.methodC();
}
}
}
也就是说ServiceB.methodB失败回滚,那么ServiceA.methodA也会回滚到savepoint点上,ServiceA.methodA可以选择另外一个分支,比如
ServiceC.methodC,继续执行,来尝试完成自己的事务。但是这个事务并没有在EJB标准中定义。
分享到:
相关推荐
自己做的一个简单的spring运用实例,主要是关于事务回滚的一个小例子,希望对还在纠结事务回滚的童靴有点帮助。。。
主要介绍了java Spring事务回滚的相关资料,需要的朋友可以参考下
花了一晚上时间终于搞出来了Spring整合Mybatis事务回滚(Mysql数据库),控制Service层中的多次插入操作,多次操作整体是一个事务。 里面有缘嘛和jar包,资源为war包,导入即可。运行Test类中的测试代码即可。 建表...
Spring/SpringMVC/MyBatis整合,可事务回滚
Spring中的JdbcTemplate,Spring中的的事务.。。。。。。。
主要给大家介绍了关于mongoDB 4.0事务回滚的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用MongoDB具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
集成了Spring 3.2.2 Hibernate 3.3.2 Struts 2.3.4 用JPA的注解实现orm 不用创建数据库表 只需本机上有oracle即可 用JSR注解进行Spring的...用Spring Test整合Junit4 进行测试并自动事务回滚 不对数据库造成持久化操作
Spring中@Transactional事务回滚(含实例详细讲解,附源码)Java开发Java经验技巧共8页.pdf.zip
本篇文章主要介绍了浅谈Spring中@Transactional事务回滚及示例(附源码),小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
主要介绍了spring事务异常回滚实例解析,具有一定借鉴价值,需要的朋友可以参考下
主要介绍了使用SpringBoot注解方式处理事务回滚实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
1. 基于Aspectj实现动态数据源...6. 实现事务内切换数据源(支持原生Spring声明式事务哟,仅此一家),并支持多数据源事务回滚(有了它除了跨服务的事务你需要考虑分布式事务,其他都不需要,极大的减少了系统的复杂程度)
Spring事务管理A方法内部调用B方法的回滚问题测试代码
spring事务的底层实现流程图 spring事务的底层实现流程图 spring事务的底层实现流程图 spring事务的底层实现流程图 spring事务的底层实现流程图 spring事务的底层实现流程图 spring事务的底层实现流程图 spring事务...
Spring的事务管理默认只对出现运行期异常(java.lang.RuntimeException及其子类)进行回滚,需要了解更多Spring事务方面的知识,可详看本
使用MySQL时,如果发现事务无法回滚,但Hibernate、Spring、JDBC等配置又没有明显问题,到底是什么原因,下面与大家分享下
主要介绍了SpringBoot事务使用及回滚实现代码详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
spring boot 纯注解方法事务控制回滚,注解+简单配置文件使用多线程demo
Spring 与 各框架的组各下来,版本就特别的多,针对简单的 使用注解来管理事务的,研究了两天,要不网上...说一下环境 Intellij Idea 15 + Maven + Spring 4.25 + Mybatis 3使用注解的方式,进行的事务回滚,欢迎测试。