Spring事务回滚实战
一、前置知识
1、Java异常中,Throwable是最顶层的父类,有Error和Exception两个子类
2、Exception分为运行时异常(RuntimeException及其子类)和非运行时异常(Exception子类中,除了RuntimeException及其子类之外的类)
3、使用spring的@Transactiona开启事务,默认Error和RuntimeException及其子类才会回滚
4、@Transactiona默认传播行为是REQUIRED,如需配合其他传播行为测试,请查看:Spring事务传播行为实战
// 回滚
@Transactional
public void add() {
KsA ksA = new KsA();
ksA.setName("林");
ksAService.insert(ksA);
throw new RuntimeException("运行报错啦");
}
// 不回滚
@Transactional
public void add() throws IOException {
KsA ksA = new KsA();
ksA.setName("林");
ksAService.insert(ksA);
throw new IOException();
}
// 不回滚
@Transactional
public void add() {
KsA ksA = new KsA();
ksA.setName("林");
ksAService.insert(ksA);
try {
throw new RuntimeException("运行报错啦");
} catch (RuntimeException e) {
e.printStackTrace();
}
}
针对跨方法捕获的异常,事务回滚,因为内部事务已经结束,确实是抛出了异常,经过AOP切面
@Transactional
public String insertForlanA(ForlanA forlanA) {
try {
forlanBService.insertForlanB(new ForlanB());
} catch (Exception e) {
e.printStackTrace();
return "特定异常结果";
}
return "成功";
}
@Transactional
public String insertForlanB(ForlanB forlanB) {
forlanBDao.insert(forlanB);
int res = 1 / 0; //java.lang.ArithmeticException: / by zero
return "成功";
}
// 回滚
@Transactional(rollbackFor = NullPointerException.class)
public void add() {
KsA ksA = new KsA();
ksA.setName("林");
ksAService.insert(ksA);
int forlan = 1 / 0;
}
这里抛出的是ArithmeticException,但我们指定是NullPointerException才回滚,为什么还是回滚了呢?
其实这两个类都是继承RuntimeException,Spring本来就默认了RuntimeException及其子类也是回滚的
我们为什么还要指定rollbackFor参数?我们来看看下面的情况
// 回滚
@Transactional(rollbackFor = IOException.class)
public void add() throws IOException {
KsA ksA = new KsA();
ksA.setName("林");
ksAService.insert(ksA);
throw new IOException();
}
我们从1.2得知,非运行时异常默认是不回滚,但我们可以通过指定rollbackFor参数来回滚结论:是针对非运行时异常的,在原基础上拓展
// 不回滚
@Transactional(noRollbackFor = ArithmeticException.class)
public void add() {
KsA ksA = new KsA();
ksA.setName("林");
ksAService.insert(ksA);
int forlan = 1 / 0;
}
// ks_a表数据插入成功,ks_b数据回滚
@Transactional
public void add() {
KsA ksA = new KsA();
ksA.setName("林");
ksAService.insert(ksA);
Object savepoint = TransactionAspectSupport.currentTransactionStatus().createSavepoint();
try {
KsB ksB = new KsB();
ksB.setAge(10);
ksBService.insert(ksB);
throw new RuntimeException("父方法报错");
} catch (Exception e) {
TransactionAspectSupport.currentTransactionStatus().rollbackToSavepoint(savepoint);
}
}