Spring事务实现误区:如何正确使用事务的传播行为
事务的传播行为是在不同类的方法中发生的!比如如下方法,该方法中事务并没有传播,始终是按照test()方法注解@Transaction注解实现的事务定义中运行的,即使事务的传播行为为Propagation.REQUIRES_NEW。
@Override
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void test() throws Exception {
test2();
test3();
}
@Override
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void test2() throws Exception {
tbTest2Dao.insertWithSql(new SQL().insert("id","context").values(0,"1"));
}
@Override
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void test3() throws Exception {
tbTest3Dao.insertWithSql(new SQL().insert("id","context").values(0,"2"));
System.out.println(1/0);
}
正确的方式如下
/**
* @author 周宁
* @Date 2019-08-08 15:24
*/
public class TestPropagationServiceImpl implements TestPropagationService{
@Override
@Transactional(propagation = Propagation.REQUIRED)
public void testService() throws Exception {
tbTest2Service.test2();
tbTest3Service.test3();
System.out.println(1/0);
}
}
public class TbTest2ServiceImpl implements TbTest2Service{
@Override
@Transactional(propagation = Propagation.REQUIRED)
public void test2() throws Exception {
tbTest2Dao.insertWithSql(new SQL().insert("id","context").values(0,"1"));
}
}
public class TbTest3ServiceImpl implements TbTest3Service {
@Override
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void test3() throws Exception {
tbTest3Dao.insertWithSql(new SQL().insert("id", "context").values(0, "2"));
}
}
这样子,在执行TbTest3ServiceImpl.test3()方法时候会重新开启新的事务执行。
Spring事务传播行为的实现原理
精华所在就是DefaultTransactionStatus类的newTransaction属性了,通过判断是否为开启的新的事务决定着事务是否提交;同一个个事务所对应的ConnectionHolder是同一个,在进行事务回滚或者提交操作时候判断当DefaultTransactionStatus的newTransaction属性是否为true如果为true,使用status中的TransactionObject对象获得到ConnectionHolder然后继续拿到connection进行回滚或者提交,如果newTransaction属性为false,设置connectionHolder的rollbackOnly属性为true,最终在进入真正开启事务的方法里面会判断ConnectionHolder的rollbackOnly属性统一的回滚。