标题 Spring声明式事物需要注意的点
先来描述一下本文是为了解决什么样的问题
有这样一个场景,在A类中有两个方法method1() 和method2() 这两个方法都是被@transaction注解修饰的,method1() 内部又调用了method2(),那么如何使method2()的事务生效呢?反应出来的问题是 一个方法中调用了本类中的另一个方法,那么内部被调用的另一个方法的事务是不会生效的。
为什么会出现不生效的问题?
Spring事务实现的原理是动态代理,无法对一个类重复代理,相当于只给这个大的方法生成了一个事务,内部调用的方法都是在一个事务里。
解决方法?
对象.方法()。导入aop依赖 @EnableAspectJAutoProxy(exposeProxy = true)暴露代理对象。
#Spring中事务的传播行为
- REQUIRED:如果有事务就和之前的事务公用一个事务,如果没有则创建一个事务
- REQUIRES_NEW:创建一个新的事务,如果以前有事务,暂停前面的事务。
- SUPPORTS:之前有事务,就用之前的事务,如果没有也可以。
- MANDATORY(强制):如果没有事务就报错。
- NOT_SUPPORTED:不支持在事务内运行,如果有事务了就挂起当前事务。
- NEVER:不支持在事务内运行,如果有就报错。
- NESTED:开启一个子事务(mysql不支持),需要支持还原点功能的数据库才行。
场景演示
内部调用的方法事务添加失败
/**
insert2 和insert3都是在insert2中的事物中,
这时候insert3的事务是不生效的
现象就是 数据库里面会出现两条记录
*/
@Override
@Transactional(rollbackFor = Exception.class)
public void insert2(User user) {
user.setName("insert2");
userMapper.insert(user);
try{
insert3(user);
}catch(Exception e){
}
}
@Override
@Transactional(propagation = Propagation.REQUIRES_NEW,rollbackFor = Exception.class)
public void insert3(User user) {
user.setName("insert3");
userMapper.insert(user);
int a = 1/0;
}
解决方法
- 倒入AOP依赖
- 启动类上添加@EnableAspectJAutoProxy(exposeProxy = true)
- 修改代码
/**
insert2和insert3在不同的事务中,
insert3会回滚,insert2不会回滚。
*/
@Override
@Transactional(rollbackFor = Exception.class)
public void insert2(User user) {
TransactionUserServiceImpl currentProxy = (TransactionUserServiceImpl) AopContext.currentProxy();
user.setName("insert2");
userMapper.insert(user);
try {
currentProxy.insert3(user);
}catch (Exception e){
e.printStackTrace();
}
}
@Override
@Transactional(propagation = Propagation.REQUIRES_NEW,rollbackFor = Exception.class)
public void insert3(User user) {
user.setName("insert3");
userMapper.insert(user);
int a = 1/0;
}
/**
insert2回滚
insert3不回滚
*/
@Override
@Transactional(rollbackFor = Exception.class)
public void insert2(User user) {
TransactionUserServiceImpl currentProxy = (TransactionUserServiceImpl) AopContext.currentProxy();
user.setName("insert2");
userMapper.insert(user);
currentProxy.insert3(user);
int a = 1/0;
}
@Override
@Transactional(propagation = Propagation.REQUIRES_NEW,rollbackFor = Exception.class)
public void insert3(User user) {
user.setName("insert3");
userMapper.insert(user);
}