一、工作基础
@Transactional 的工作机制是基于 AOP 实现,AOP 是使用动态代理实现的。
二、基本属性
三、propagation的详细参数
- REQUIRED:如果有事务, 那么加入事务, 没有的话新建一个(默认情况下)
- NOT_SUPPORTED:容器不为这个方法开启事务
- REQUIRES_NEW:不管是否存在事务,都创建一个新的事务,原来的挂起,新的执行完毕,继续执行老的事务
- MANDATORY:必须在一个已有的事务中执行,否则抛出异常
- NEVER:必须在一个没有的事务中执行,否则抛出异常(与MANDATORY相反)
- SUPPORTS:如果其他bean调用这个方法,在其他bean中声明事务,那就用事务.如果其他bean没有声明事务,那就不用事务.
- NESTED: 如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则进行与PROPAGATION_REQUIRED类似的操作。
四、isolation的详细参数
- READ_UNCOMMITTED:读取未提交数据(会出现脏读, 不可重复读) 基本不使用
- READ_COMMITTED:读取已提交数据(会出现不可重复读和幻读)
- REPEATABLE_READ:可重复读(会出现幻读)
- SERIALIZABLE:串行化
五、@Transactional 的使用范围
@Transactional 加于private方法, 无效
@Transactional 加于未加入接口的public方法, 再通过普通接口方法调用, 无效
@Transactional 加于接口方法, 无论下面调用的是private或public方法, 都有效(timeout放在下面调用的private或public方法上都不生效,只在外层方法中有效)@Transactional 加于接口方法后, 被本类普通接口方法直接调用, 无效
@Transactional 加于接口方法后, 被本类普通接口方法通过接口调用, 有效
@Transactional 加于接口方法后, 被它类的接口方法调用, 有效
@Transactional 加于接口方法后, 被它类的私有方法调用后, 有效
六、解决本次使用中遇到的问题:加于接口方法后, 被本类普通接口方法直接调用, 无效
出现这个问题的主要原因就是@Transactional 的工作机制是基于 AOP 实现,直接在本类调用,没有经过AOP代理,因此无论是直接调用还是this调用都不会生效。
解决办法:把所在类手动接入AOP代理
具体方法:
1、使用@Autowired注解,把自己加入AOP代理,
@Autowired
private TestService testService;
testService.a();//a方法为内部方法
2、使用AopContext
private DelivChildProjectController currentService() { return ((DelivChildProjectController) AopContext.currentProxy()); }