对Spring事务一些问题的讨论
提起spring事务,就会让人联想起四大基本特征,五个隔离级别,七大传播特性。相信大多数人都知道这些东西,但是知道是一回事情,能用好真的是另一回事了。在使用Spring事务的时候,我曾遇到过几个比较严肃的问题,在这里我做一个自我总结。
问题一、 propagation.NESTED和propagation.REQUIRED_NEW有什么区别?
当调用方不存在事务的时候,两者的效果是一致的。所以这里讨论问题的前提是调用方存在事务。PROPAGATION_REQUIRES_NEW 启动一个新的, 不依赖于环境的 "内部" 事务. 这个事务将被完全 commited 或 rolled back 而不依赖于外部事务, 它拥有自己的隔离范围, 自己的锁, 等等. 当内部事务开始执行时, 外部事务将被挂起, 内务事务结束时, 外部事务将继续执行.
另一方面, PROPAGATION_NESTED 开始一个 "嵌套的" 事务, 它是已经存在事务的一个真正的子事务. 潜套事务开始执行时, 它将取得一个 savepoint. 如果这个嵌套事务失败, 我们将回滚到此 savepoint. 潜套事务是外部事务的一部分, 只有外部事务结束后它才会被提交.
由此可见, PROPAGATION_REQUIRES_NEW 和 PROPAGATION_NESTED 的最大区别在于, PROPAGATION_REQUIRES_NEW 完全是一个新的事务, 而 PROPAGATION_NESTED 则是外部事务的子事务, 如果外部事务 commit, 嵌套事务也会被 commit, 这个规则同样适用于 roll back.
问题二、 @Transactional为什么会失效?
1.调用方和被调用方属于同一个component,被调用方的 @Transacational注解无效
package com.transacational; import org.springframework.stereotype.Component;
import org.springframework.transaction.annotation.Transactional; /**
* Created by chenqimiao on 17/10/31.
*/
@Component
public class Service { public void test1(){
test2();
} @Transactional//此处的注解无效
public void test2(){ }
}
2.被调用方不是一个public方法,被调用方的 @Transacational注解无效
@Component
public class Service { @Resource
private Service1 service1; public void test1(){
test2();
service1.test3();
} @Transactional//1.此处的注解无效
public void test2(){ }
}
package com.transacational; import org.springframework.stereotype.Component;
import org.springframework.transaction.annotation.Transactional; /**
* Created by chenqimiao on 17/10/31.
*/
@Component
public class Service1 { @Transactional//2.此处注解无效
protected void test3(){ }
}
3.未开启事务开关,如:在SpringBoot中,启动类未使用 @EnableTransactionManagement
问题三、 如何理解@Transactional的超时时间?
timeout是一个供开发者设置超时时间的属性。默认值-1,超时时间由具体的sql系统决定。
/**
* Created by chenqimiao on 17/10/31.
*/
@Component
public class Service3 { @Resource
private AdminInfoDoMapper adminInfoDoMapper;
@Transactional(timeout = 4)//并不会超时
public void test4(){ adminInfoDoMapper.selectNameById(1);
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
超时时间具体的定义:事务开始(在该方法第一句代码执行之前)到最后一个Statement执行完毕
所以象下面这样写,事务就会超时
@Component
public class Service3 { @Resource
private AdminInfoDoMapper adminInfoDoMapper;
@Transactional(timeout = 4)
public void test4(){
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
adminInfoDoMapper.selectNameById(1); }
}
问题四、 @Transactional默认的回滚策略?
默认情况下,只有当RuntimeException或其子类的异常被事务捕获之后,事务才会回滚,如果要让事务能够回滚所有异常,必须手动指定 @Transactional(rollbackFor=Exception.class) ,这样继承Exception的子类或者Exception本身都可以让事务回滚。
最新文章
- 10. JEB1.5 插件编写二
- 转:ASP.NET 使用Ajax
- sphinx配置文件sphinx.conf参数详细说明
- CocurrentHashMap和Hashtable的区别
- HDU 3555 数位dp入门
- jquery()的三种$()
- Android NDK 开发(三)--常见错误锦集合Log的使用【转】
- Create,Insert
- Maximum number of WAL files in the pg_xlog directory (2)
- 如何搭建Struts2环境
- 部署K2 Blackpearl流程时出错(与基础事务管理器的通信失败或Communication with the underlying transaction manager has failed.
- 类 ArrayBlockingQueue<;E>;(一个由数组支持的有界阻塞队列。)
- linux-telnet服务配置
- UIButton 头文件常见属性和方法
- js判断undefined类型,undefined,null,NaN的区别
- redhat在线安装chrome浏览器
- Luogu [USACO08OPEN]寻宝之路Clear And Present Danger
- Python元祖
- System.nanoTime与System.currentTimeMillis的区别(转)
- File的创建