一、部分内容是收集来的。
默认spring事务只在发生未被捕获的 runtimeexcetpion时才回滚。
spring aop 异常捕获原理:被拦截的方法需显式抛出异常,并不能经任何处理,这样aop代理才能捕获到方法的异常,才能
进行回滚,默认情况下aop只捕获runtimeexception的异常,但可以通过
配置来捕获特定的异常并回滚
换句话说在service的方法中不使用try catch 或者在catch中最后加上throw new runtimeexcetpion(),这样程序异常时才
能被aop捕获进而回滚
解决方案:
方案1.例如service层处理事务,那么service中的方法中不做异常捕获,或者在catch语句中最后增加throw new
RuntimeException()语句,以便让aop捕获异常再去回滚,并且在service上层(webservice客户端,view层action)要继续捕获
这个异常并处理
方案2.在service层方法的catch语句中增加:TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();语句,手动回滚,这样上层就无需去处理异常
2020年5月29日
//JdbcTemplateTest类
@Transactional
public void getBoook(){
int update = jdbcTemplate.update("insert into user (name) values ('zzz')");
System.out.println(update);
updateBook();
}
public void updateBook(){
int update = jdbcTemplate.update("insert into user (name) values ('list')");
int i = 1/0;
jdbcTemplate.update("insert into salary (salary,userId) values (10.0,2)");
}
Test
public void test02(){
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
JdbcTemplateTest bean = applicationContext.getBean(JdbcTemplateTest.class);
bean.getBoook();
}
结果:事务生效,数据库两张表都没有插入数据
public void getBoook(){
int update = jdbcTemplate.update("insert into user (name) values ('zzz')");
System.out.println(update);
updateBook();
}
@Transactional
public void updateBook(){
int update = jdbcTemplate.update("insert into user (name) values ('list')");
int i = 1/0;
jdbcTemplate.update("insert into salary (salary,userId) values (10.0,2)");
}
@Test
public void test02(){
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
JdbcTemplateTest bean = applicationContext.getBean(JdbcTemplateTest.class);
bean.getBoook();
}
结果:没有事务,user表插入两条数据。
结论:事务是基于aop原理实现得 当调用getBook方法时切面类已经起作用,即使这个方法再调用其他带有Transactional得注解,也不会再去添加事务。因为这里是本类的方法调用本类的方法,根据aop的原理可以看出,是目标类的方法(getBook)去调用目标类的方法(updateBook),此过程并没有代理类的事,所以没有增强效果(事务)。