使用spring不免要用到spring的事务管理,要用事务管理又会很天然的选择声明式的事务管理,在spring的文档中说道,spring声明式事务管理默认对非检查型异常和运行时异常进行事务回滚,而对检查型异常则不进行回滚操做。
那么什么是检查型异常什么又是非检查型异常呢?
最简单的判断点有两个:
1.继承自runtimeexception或error的是非检查型异常,而继承自exception的则是检查型异常(固然,runtimeexception自己也是exception的子类)。java
2.对非检查型类异常能够不用捕获,而检查型异常则必须用try语句块进行处理或者把异常交给上级方法处理总之就是必须写代码处理它。因此必须在service捕获异常,而后再次抛出,这样事务方才起效。spring
在service类前加上@Transactional,声明这个service全部方法须要事务管理。每个业务方法开始时都会打开一个事务。Spring默认状况下会对运行期例外(RunTimeException)进行事务回滚。这个例外是unchecked若是遇到checked意外就不回滚。spa
若是在程序须要捕获异常 ,则可:代理
try { doDb1(); doDb2(); } catch (Exception e) { e.printStackTrace(); TransactionAspectSupport.currentTransactionStatus().setRollbackOnly(); //若是doDb2()抛了异常,doDb1()是会回滚的 }
在catch块中添加TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();或者再往外抛try{}catch{throw Exception}。code
在须要事务管理的地方加@Transactional 注解。@Transactional 注解能够被应用于接口定义和接口方法、类定义和类的 public 方法上。对象
@Transactional 注解只能应用到 public 可见度的方法上。 若是你在 protected、private 或者 package-visible 的方法上使用 @Transactional 注解,它也不会报错, 可是这个被注解的方法将不会展现已配置的事务设置。继承
注意仅仅 @Transactional 注解的出现不足于开启事务行为,它仅仅 是一种元数据。必须在配置文件中使用配置元素,才真正开启了事务行为。接口
经过 元素的 “proxy-target-class” 属性值来控制是基于接口的仍是基于类的代理被建立。若是 “proxy-target-class” 属值被设置为 “true”,那么基于类的代理将起做用(这时须要CGLIB库cglib.jar在CLASSPATH中)。若是 “proxy-target-class” 属值被设置为 “false” 或者这个属性被省略,那么标准的JDK基于接口的代理将起做用。事务
Spring团队建议在具体的类(或类的方法)上使用 @Transactional 注解,而不要使用在类所要实现的任何接口上。在接口上使用 @Transactional 注解,只能当你设置了基于接口的代理时它才生效。由于注解是 不能继承 的,这就意味着若是正在使用基于类的代理时,那么事务的设置将不能被基于类的代理所识别,并且对象也将不会被事务代理所包装。文档
@Transactional 的事务开启 ,或者是基于接口的 或者是基于类的代理被建立。因此在同一个类中一个方法调用另外一个方法有事务的方法,事务是不会起做用的。