1、不建议在接口上添加@Transactional注解,一般在service类标签上添加@Transactional即可
2、@Transactional注解只能应用到public可见度的方法上。如果应用到protected、private或者package可见度的方法上时,不会报错,但事务也不会起作用
3、默认情况下,spring会对uncheck异常进行事务回滚的;如果是checked异常则不会回滚,可添加注解 @Transactional(rollbackFor=Exception.class) 是的checked异常回滚。
uncheck异常:java里面将派生于Error或者RuntimeException(比如空指针,1/0)的异常
checked异常:其他继承自java.lang.Exception得异常统称为Checked Exception,如IOException、TimeoutException等
4、数据库引擎需要支持事务管理,如果是mysql,注意表要使用事务的引擎,比如innodb,如果是myisam,事务不会起作用的
5、同一类中methodA()方法没有@Transactional 注解,在其内部调用有@Transactional 注解的方法,有@Transactional 注解的方法methodB()的事务被忽略,不会开启新的事务,也不会发生回滚。例如:
@Service
public class TransactionService {
public void methodA(){
this.methodB();
}
@Transactional
public void methodB(){
}
}
原因: Spring采用动态代理(AOP)实现对bean的管理和切片,它为我们的每个class生成一个代理对象。只有在代理对象之间进行调用时,可以触发切面逻辑。在 Spring 的 AOP 代理下,只有目标方法由外部调用,目标方法才由 Spring 生成的代理对象来管理。
详解: Spring的事务管理是通过AOP实现的,其AOP的实现对于非final类是通过cglib这种方式,即生成当前类的一个子类作为代理类,然后在调用其下的方法时,会判断这个方法有没有@Transactional注解,如果有的话,则会开启一个新的事务,并通过动态代理实现事务管理(拦截方法调用,执行事务等切面)。当methodA()中调用methodB()时,并不是使用的代理对象,而是普通的javabean,从而导致this.methodB()时也不是代码对象,从而导致@Transactional失败,即发现methodA()上并没有@Transactional注解,所以整个AOP代理过程(事务管理)不会发生。
解决办法:
1、把这两个方法分开到不同的类中;
2、把注解@Transactional加到类名上面去;
3、把注解@Transactional加到methodA()方法上,methodB()不添加注解,在调用methodB()时两个方法的事务都会生效,因为methodA()默认的事务传播属性为PROPAGATION_REQUIRED,此时methodB()会加入到methodA()中
@Service
public class TransactionService {
@Transactional
public void methodA(){
this.methodB();
}
public void methodB(){
}
}
4、获取本对象的代理对象,再进行调用。具体操作如:
- Spring-content.xml上下文中,增加配置:<aop:aspectj-autoproxy expose-proxy="true"/>
- 在TransactionService 中,用(transactionService )(AopContext.currentProxy()),获取到TransactionService 的代理类,再调用事务方法,强行经过代理类,激活事务切面。
网友评论