spring事务注解 @Transactional

Advantages of the Spring Framework’s Transaction Support Model :: Spring Framework

Propagation

REQUIRED(TransactionDefinition.PROPAGATION_REQUIRED)

   默认值 如果当前事务不存在,则创建新的,如果存在则加入

SUPPORTS(TransactionDefinition.PROPAGATION_SUPPORTS)

   支持当前事务,如果不存在则以非事务运行

MANDATORY(TransactionDefinition.PROPAGATION_MANDATORY)

   支持当前事务,如果不存在则抛出异常

REQUIRES_NEW(TransactionDefinition.PROPAGATION_REQUIRES_NEW)

  如果当前有事务,则挂起该事务,并且自己创建一个新的事务给自己使用   如果当前没有事务,则同REQUIRED

NOT_SUPPORTED(TransactionDefinition.PROPAGATION_NOT_SUPPORTED),

  如果当前有事务,则把事务挂起,自己不适用事务去运行数据库操

NEVER(TransactionDefinition.PROPAGATION_NEVER),

  如果当前有事务存在,则抛出异常

NESTED(TransactionDefinition.PROPAGATION_NESTED);

  1. 如果当前有事务,则开启子事务(嵌套事务),嵌套事务是独立提交或者回滚;
  2. 如果当前没有事务,则同REQUIRED。但是如果主事务提交,则会携带子事务一起提交。
  3. 如果主事务回滚,则子事务会一起回滚。相反,子事务异常,则父事务可以回滚或不回滚

事务不生效

不是public

@Service 
public class UserService { 
     
    @Transactional 
    private void add(UserModel userModel) { 
         saveData(userModel); 
         updateData(userModel); 
    } 
}

spring要求被代理方法必须是public的

 在AbstractFallbackTransactionAttributeSource类的computeTransactionAttribute方法中有个判断,如果目标方法不是public,则TransactionAttribute返回null,即不支持事务。

protected TransactionAttribute computeTransactionAttribute(Method method, @Nullable Class<?> targetClass) { 
    // Don't allow no-public methods as required. 
  if (allowPublicMethodsOnly() && !Modifier.isPublic(method.getModifiers())) { 
       return null; 
    } 
    Method specificMethod = AopUtils.getMostSpecificMethod(method, targetClass); 
 
    // First try is the method in the target class. 
    TransactionAttribute txAttr = findTransactionAttribute(specificMethod); 
    if (txAttr != null) { 
      return txAttr; 
    } 

方法用final修饰

spring事务底层使用了aop,也就是通过jdk动态代理或者cglib,帮我们生成了代理类,在代理类中实现的事务功能。

但如果某个方法用final修饰了,那么在它的代理类中,就无法重写该方法,而无法添加事务功能。

 方法是static

同样无法通过动态代理,变成事务方法。

 方法内部调用

// 现象
@Service
public class OrderServiceImpl implements OrderService {
 
    public void update(Order order) {
        updateOrder(order);
    }
    @Transactional
    public void updateOrder(Order order) {
        // update order
    }

//解决

方法一:新加一个Service方法
新加一个Service方法,把@Transactional注解加到新Service方法上,把需要事务执行的代码移到新方法中
@Servcie 
public class ServiceA { 
   @Autowired 
   prvate ServiceB serviceB; 
 
   public void save(User user) { 
         queryData1(); 
         queryData2(); 
         serviceB.doSave(user); 
   } 
 } 
 
 @Servcie 
 public class ServiceB { 
 
 @Transactional(rollbackFor=Exception.class) 
    public void doSave(User user) { 
       addData1(); 
       updateData2(); 
    } 
 }


方法二:AopContent
@Servcie 
public class ServiceA { 
 
   public void save(User user) { 
         queryData1(); 
         queryData2(); 
  ((ServiceA)AopContext.currentProxy()).doSave(user); 
   } 
 
   @Transactional(rollbackFor=Exception.class) 
   public void doSave(User user) { 
       addData1(); 
       updateData2(); 
    } 
 }

方法三:在该Service类中注入自己
@Servcie 
public class ServiceA { 
   @Autowired 
prvate ServiceA serviceA; 
 
   public void save(User user) { 
         queryData1(); 
         queryData2(); 
         serviceA.doSave(user); 
   } 
 
   @Transactional(rollbackFor=Exception.class) 
   public void doSave(User user) { 
       addData1(); 
       updateData2(); 
    } 
 }

 未被spring容器管理

使用spring事务的前提是:对象要被spring管理,需要创建bean实例。

// @Service
public class OrderServiceImpl implements OrderService {
 
    @Transactional
    public void updateOrder(Order order) {
         // update order
    }
}

 多线程调用

@Slf4j 
@Service 
public class UserService { 
 
    @Autowired 
    private UserMapper userMapper; 
    @Autowired 
    private RoleService roleService; 
 
    @Transactional 
    public void add(UserModel userModel) throws Exception { 
        userMapper.insertUser(userModel); 
        new Thread(() -> { 
            roleService.doOtherThing(); 
        }).start(); 
    } 
} 
 
@Service 
public class RoleService { 
 
    @Transactional 
    public void doOtherThing() { 
        System.out.println("保存role表数据"); 
    } 
}

 从上面的例子中,我们可以看到事务方法add中,调用了事务方法doOtherThing,但是事务方法doOtherThing是在另外一个线程中调用的。

这样会导致两个方法不在同一个线程中,获取到的数据库连接不一样,从而是两个不同的事务。如果想doOtherThing方法中抛了异常,add方法也回滚是不可能的

spring的事务是通过数据库连接来实现的。当前线程中保存了一个map,key是数据源,value是数据库连接。

private static final ThreadLocal<Map<Object, Object>> resources =
  new NamedThreadLocal<>("Transactional resources");

我们说的同一个事务,其实是指同一个数据库连接,只有拥有同一个数据库连接才能同时提交和回滚。如果在不同的线程,拿到的数据库连接肯定是不一样的,所以是不同的事务。

手动控制事务(lock)

在事务中,使用了Redis分布式锁.这个方法一旦执行,事务生效,接着就Redis分布式锁生效,代码执行完后,先释放Redis分布式锁,然后再提交事务数据,最后事务结束。在这个过程中,事务没有提交之前,分布式锁已经被释放,导致分布式锁失效

这是因为:

springAop,会在updateDB方法之前开启事务,之后再加锁,当锁住的代码执行完成后,再提交事务,因此锁住的代码块执行是在事务之内执行的,可以推断在代码块执行完时,事务还未提交,锁已经被释放,此时其他线程拿到锁之后进行锁住的代码块,读取的库存数据不是最新的

还没有开事务之前就加锁,那么就可以保证线程的安全性

@Autowired
PlatformTransactionManager platformTransactionManager;

@Autowired
TransactionDefinition  transactionDefinition;
@Autowired
SysUserMapper sysUserMapper;
@Test
public void sqlSel(){

   ReentrantLock reentrantLock = new ReentrantLock();
   TransactionStatus transaction = null;
   try {
       if (reentrantLock.tryLock(20, TimeUnit.SECONDS)){

           transaction = platformTransactionManager.getTransaction(transactionDefinition);
           SysUser sysUser = sysUserMapper.selectByPrimaryKey(1l);
           sysUser.setUserName("Transaction");
           sysUserMapper.updateByPrimaryKey(sysUser);
       }else{
           System.out.println("Ss");
       }
   }catch (Exception e){
       if (transaction != null) {
           platformTransactionManager.rollback(transaction);
       }

       e.printStackTrace();
   }finally {
       if (transaction != null) {
           platformTransactionManager.commit(transaction);
       }
       reentrantLock.unlock();
   }
}

事务不回滚

 错误的传播特性

Propagation.NOT_SUPPORTED,这种类型的传播特性不支持事务,如果有事务则会抛异常。

目前只有这三种传播特性才会创建新事务:NESTED,REQUIRES_NEW,REQUIRED。

@Service 
public class OrderServiceImpl implements OrderService{ 
 
    @Transactional
    public void update(Order order) {
        updateOrder(order);
    }
 
    @Transactional(propagation = Propagation.NOT_SUPPORTED)
    public void updateOrder(Order order) {
        // update order
    }

自己吞了异常

把异常吃了,然后又不抛出来,事务也不会回滚!

@Transactional
public void CatchExceptionCanNotRollback() {

    try {
        extraAdDao.save(new ExtraAd("qinyi"));
        throw new RuntimeException();
    } catch (Exception ex) {
        ex.printStackTrace();
        // 手动标记回滚
      TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
    }
}

 手动抛了别的异常

即使开发者没有手动捕获异常,但如果抛的异常不正确,spring事务也不会回滚。

@Service
public class OrderServiceImpl implements OrderService {
 
    @Transactional
    public void updateOrder(Order order) {
        try {
            // update order
        } catch {
            throw new Exception("更新错误");
        }
    }
}

这样事务也是不生效的,因为默认回滚的是:RuntimeException,如果你想触发其他异常的回滚,需要在注解上配置一下,如:@Transactional(rollbackFor = Exception.class), 这个配置仅限于 Throwable 异常类及其子类

  自定义了回滚异常

@Slf4j 
@Service 
public class UserService { 
     
    @Transactional(rollbackFor = BusinessException.class) 
    public void add(UserModel userModel) throws Exception { 
       saveData(userModel); 
       updateData(userModel); 
    } 
}

保存和更新数据时,程序报错了,抛了SqlException、DuplicateKeyException等异常。而BusinessException是我们自定义的异常,报错的异常不属于BusinessException,所以事务也不会回滚。

即使rollbackFor有默认值,但阿里巴巴开发者规范中,还是要求开发者重新指定该参数。

因为如果使用默认值,一旦程序抛出了Exception,事务不会回滚,这会出现很大的bug。所以,建议一般情况下,将该参数设置成:Exception或Throwable

捕捉异常并转换异常, 导致不能回滚

@Override
@Transactional
public void NotRuntimeExceptionCanNotRollback() throws CustomException {

    try {
        extraAdDao.save(new ExtraAd("qinyi"));
        throw new RuntimeException();
    } catch (Exception ex) {
        throw new CustomException(ex.getMessage());
    }
}
public class CustomException extends Exception {

    public CustomException(String message) {
        super(message);
    }
}

指定异常, 可以回滚

@Override
@Transactional(rollbackFor = {CustomException.class})
public void AssignExceptionCanRollback() throws CustomException {

    try {
        extraAdDao.save(new ExtraAd("qinyi"));
        throw new RuntimeException();
    } catch (Exception ex) {
        throw new CustomException(ex.getMessage());
    }
}

 嵌套事务回滚多了

public class UserService {
 
    @Autowired 
    private UserMapper userMapper;
 
    @Autowired 
    private RoleService roleService;
 
    @Transactional 
    public void add(UserModel userModel) throws Exception {
        userMapper.insertUser(userModel);
        roleService.doOtherThing();
    }
}
 
@Service 
public class RoleService {
 
    @Transactional(propagation = Propagation.NESTED)
    public void doOtherThing() {
        System.out.println("保存role表数据");
    }
}

可以将内部嵌套事务放在try/catch中,并且不继续往上抛异常。这样就能保证,如果内部嵌套事务中出现异常,只回滚内部事务,而不影响外部事务。

@Slf4j 
@Service 
public class UserService { 
 
    @Autowired 
    private UserMapper userMapper; 
 
    @Autowired 
    private RoleService roleService; 
 
    @Transactional 
    public void add(UserModel userModel) throws Exception { 
 
        userMapper.insertUser(userModel); 
        try { 
            roleService.doOtherThing(); 
        } catch (Exception e) { 
            log.error(e.getMessage(), e); 
        } 
    } 
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值