Spring 事务管理

什么是事务

事务是逻辑上的一组操作,要么都执行,要么都不执行。
事务能否生效数据库引擎是否支持事务是关键。比如常用的 MySQL 数据库默认使用支持事务的innodb引擎。但是,如果把数据库引擎变为 myisam,那么程序也就不再支持事务了

事务的特性(ACID)

  • 原子性(Atomicity): 一个事务(transaction)中的所有操作,或者全部完成,或者全部不完成,不会结束在中间某个环节。事务在执行过程中发生错误,会被回滚(Rollback)到事务开始前的状态,就像这个事务从来没有执行过一样。即,事务不可分割、不可约简
  • 一致性(Consistency): 在事务开始之前和事务结束以后,数据库的完整性没有被破坏。这表示写入的资料必须完全符合所有的预设约束、触发器、级联回滚等。
  • 隔离性(Isolation): 数据库允许多个并发事务同时对其数据进行读写和修改的能力,隔离性可以防止多个事务并发执行时由于交叉执行而导致数据的不一致。事务隔离分为不同级别,包括未提交读(Read uncommitted)、提交读(read committed)、可重复读(repeatable read)和串行化(Serializable)。
  • 持久性(Durability): 事务处理结束后,对数据的修改就是永久的,即便系统故障也不会丢失。

MySQL 怎么保证原子性的
我们知道如果想要保证事务的原子性,就需要在异常发生时,对已经执行的操作进行回滚,在 MySQL 中,恢复机制是通过 回滚日志(undo log) 实现的,所有事务进行的修改都会先记录到这个回滚日志中,然后再执行相关的操作。如果执行过程中遇到异常的话,我们直接利用 回滚日志 中的信息将数据回滚到修改之前的样子即可!并且,回滚日志会先于数据持久化到磁盘上。这样就保证了即使遇到数据库突然宕机等情况,当用户再次启动数据库的时候,数据库还能够通过查询回滚日志来回滚将之前未完成的事务。

事务实现方式

在Spring中,事务有两种实现方式,分别是编程式事务管理和声明式事务管理两种方式。

  • 编程式事务管理: 编程式事务管理使用TransactionTemplate或者直接使用底层的PlatformTransactionManager。对于编程式事务管理,spring推荐使用TransactionTemplate。
  • 声明式事务管理: 建立在AOP之上的。其本质是对方法前后进行拦截,然后在目标方法开始之前创建或者加入一个事务,在执行完目标方法之后根据执行情况提交或者回滚事务。

编程式事务是指在代码中手动的管理事务的提交、回滚等操作,代码侵入性比较强。声明式事务基于AOP面向切面的,它将具体业务与事务处理部分解耦,代码侵入性很低,所以在实际开发中声明式事务用的比较多。声明式事务也有两种实现方式,一是基于TXAOP的xml配置文件方式,二种就是基于@Transactional注解 声明式事务控制粒度比较粗,最小也是方法级别的,编程式事务可以控制到代码块级别

声明式事务 @Transactional
@Transactional可以用于接口,类和类方法上

  • 作用于类上时:该类的所有 public 方法将都具有该类型的事务属性
  • 作用于方法:当类配置了@Transactional,方法也配置了@Transactional,方法的事务会覆盖类的事务配置信息
  • 作用于接口:不推荐这种使用方法,因为一旦标注在Interface上并且配置了Spring AOP 使用CGLib动态代理,将会导致@Transactional注解失效
@Transactional(rollbackFor = Exception.class, propagation = Propagation.REQUIRES_NEW)
public void addStudent(StudentAdd studentAdd) {
    Student student = new Student();
    student.setAge(studentAdd.getAge());
    student.setName(studentAdd.getName());
    student.setTeam(studentAdd.getTeam());

    mapper.insert(student);
    mapper.updateById(student1);
}
@Transactional注解的属性
  1. propagation 代表事务的传播行为,默认值为 Propagation.REQUIRED,其他的属性信息如下:

    • Propagation.REQUIRED:如果当前存在事务,则加入该事务,如果当前不存在事务,则创建一个新的事务。( 也就是说如果A方法和B方法都添加了注解,在默认传播模式下,A方法内部调用B方法,会把两个方法的事务合并为一个事务
    • Propagation.SUPPORTS:如果当前存在事务,则加入该事务;如果当前不存在事务,则以非事务的方式继续运行。
    • Propagation.MANDATORY:如果当前存在事务,则加入该事务;如果当前不存在事务,则抛出异常。
    • Propagation.REQUIRES_NEW:重新创建一个新的事务,如果当前存在事务,暂停当前的事务。( 当类A中的 a 方法用默认Propagation.REQUIRED模式,类B中的 b方法加上采用 Propagation.REQUIRES_NEW模式,然后在 a 方法中调用 b方法操作数据库,然而 a方法抛出异常后,b方法并没有进行回滚,因为Propagation.REQUIRES_NEW会暂停 a方法的事务 )
    • Propagation.NOT_SUPPORTED:以非事务的方式运行,如果当前存在事务,暂停当前的事务
    • Propagation.NEVER:以非事务的方式运行,如果当前存在事务,则抛出异常。
    • Propagation.NESTED :如果当前存在事务,则创建 一个事务作为当前事务的嵌套事务来运行,如果当前没有事务,则该取值等价于:Propagation.REQUIRED
  2. isolation事务的隔离级别,默认值为 Isolation.DEFAULT

    • TransactionDefinition.ISOLATION_DEFAULT: 默认使用数据库的隔离级别,Mysql 默认采用的 REPEATABLE_READ隔离级别 Oracle 默认采用的 READ_COMMITTED隔离级别.
    • TransactionDefinition.ISOLATION_READ_UNCOMMITED: 最低的隔离级别,允许读取未提交的数据变更,会导致脏读,不可重复读,幻读。
    • TransactionDefinition.ISOLATION_READ_COMMITED: 允许读取并发事务已经提交的数据,会导致,不可重复读,幻读。
    • TransactionDefinition.ISOLATION_REPEATABLE_READ: 对同一字段的多次读取结果都是一致的,除非被本身事务自己所修改,会导致,幻读。
    • TransactionDefinition.ISOLATION_SERIALIZABLE: 最高的隔离级别,完全服从ACID的隔离级别,所有事务依次执行,这样事务之间就不会产生干扰,但是会严重影响程序性能。
  3. timeout 事务超时时间

    • timeout :事务的超时时间,默认值为 -1。如果超过该时间限制但事务还没有完成,则自动回滚事务。
  4. readOnly 属性

    • readOnly:指定事务是否为只读事务,默认值为 false;为了忽略那些不需要事务的方法,比如读取数据,可以设置 read-only 为 true。
    • 如果你一次执行多条查询语句,例如统计查询,报表查询,在这种场景下,多条查询 SQL 必须保证整体的读一致性,否则,在前条 SQL 查询之后,后条 SQL 查询之前,数据被其他用户改变,则该次整体的统计查询将会出现读数据不一致的状态,此时,应该启用事务支持
  5. rollbackFor 属性

    • rollbackFor :用于指定能够触发事务回滚的异常类型,可以指定多个异常类型。
  6. noRollbackFor属性

    • noRollbackFor:抛出指定的异常类型,不回滚事务,也可以指定多个异常类型。
事务失效场景

java开发经常使用到@Transactional注解控制事务,比较方便,但是有时候莫名的失效了,下面说下一些失效的场景:
1. @Transactional 应用在非 public 修饰的方法上

如果Transactional注解应用在非public 修饰的方法上,Transactional将会失效

之所以会失效是因为在Spring AOP 代理时, TransactionInterceptor (事务拦截器)在目标方法执行前后进行拦截,DynamicAdvisedInterceptor(CglibAopProxy 的内部类)的 intercept 方法或 JdkDynamicAopProxy 的 invoke 方法会间接调用 AbstractFallbackTransactionAttributeSourcecomputeTransactionAttribute 方法,获取Transactional 注解的事务配置信息。

protected TransactionAttribute computeTransactionAttribute(Method method,
    Class<?> targetClass) {
        // Don't allow no-public methods as required.
        if (allowPublicMethodsOnly() && !Modifier.isPublic(method.getModifiers())) {
        return null;
}

此方法会检查目标方法的修饰符是否为 public,不是 public则不会获取@Transactional 的属性配置信息。
2. @Transactional 注解属性 propagation 设置错误

这种失效是由于配置错误,若是错误的配置以下三种 propagation,事务将不会发生回滚。

TransactionDefinition.PROPAGATION_SUPPORTS:如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行。
TransactionDefinition.PROPAGATION_NOT_SUPPORTED:以非事务方式运行,如果当前存在事务,则把当前事务挂起。
TransactionDefinition.PROPAGATION_NEVER:以非事务方式运行,如果当前存在事务,则抛出异常。
3. @Transactional 注解属性 rollbackFor 设置错误

rollbackFor 可以指定能够触发事务回滚的异常类型。Spring默认抛出了未检查unchecked异常(继承自 RuntimeException的异常)或者 Error才回滚事务;其他异常不会触发回滚事务。如果在事务中抛出其他类型的异常,但却期望 Spring 能够回滚事务,就需要指定 rollbackFor属性。
在这里插入图片描述
4. 同一个类中方法调用,导致@Transactional失效

开发中避免不了会对同一个类里面的方法调用,比如有一个类Test,它的一个方法A,A再调用本类的方法B(不论方法B是用public还是private修饰),但方法A没有声明注解事务,而B方法有。则外部调用方法A之后,方法B的事务是不会起作用的。这也是经常犯错误的一个地方。

那为啥会出现这种情况?其实这还是由于使用Spring AOP代理造成的,因为只有当事务方法被当前类以外的代码调用时,才会由Spring生成的代理对象来管理。
还有一种情况,同一个类中的事务A方法调用事务B方法,B方法的事务同样不起作用

解决方式:可以通过拆分类调用方法,这样就能通过代理的方式调用。还可以通过AopContext.currentProxy()来获取代理调用,springboot中如果使用AopContext需要在启动类添加@EnableAspectJAutoProxy 注解
代码示例:

@Transactional(rollbackFor = Exception.class)
@Override
public void addStudent(StudentAdd studentAdd) {
    Student student = new Student();
    student.setAge(studentAdd.getAge());
    student.setName(studentAdd.getName());
    student.setTeam(studentAdd.getTeam());

    mapper.insert(student);

    ((StudentServiceImpl)AopContext.currentProxy()).updateStudent();

    if (true) {
        throw new RuntimeException("测试事务");
    }

}
@Transactional(rollbackFor = Exception.class, propagation = Propagation.REQUIRES_NEW)
public void updateStudent() {
    Student student1 = new Student();
    student1.setId(12);
    student1.setName("h23");
    mapper.updateById(student1);
}

5. 异常被 catch“吃了”导致@Transactional失效

@Transactional(rollbackFor = Exception.class)
public void addStudent(StudentAdd studentAdd) {
    try {
        Student student = new Student();
        student.setAge(studentAdd.getAge());
        student.setName(studentAdd.getName());
        student.setTeam(studentAdd.getTeam());

        mapper.insert(student);

    	((StudentServiceImpl)AopContext.currentProxy()).updateStudent();
    } catch(){
        e.printStackTrace();
    }
}

如果updateStudent()方法抛出了异常,addStudent不会进行回滚,除非在catch中抛出异常或TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();进行手动回滚操作
使用Object savePoint = TransactionAspectSupport.currentTransactionStatus().createSavepoint();
设置回滚点,使用TransactionAspectSupport.currentTransactionStatus().rollbackToSavepoint(savePoint);回滚到savePoint

6. 数据库引擎不支持事务
这种情况出现的概率并不高,事务能否生效数据库引擎是否支持事务是关键。常用的MySQL数据库默认使用支持事务的innodb引擎。一旦数据库引擎切换成不支持事务的myisam,那事务就从根本上失效了

编程式事务

相对来说编程式事务用到的比较少,这里简单说下怎么用

代码示例:

@Service
public class StudentUpdateService {
    @Autowired
    private StudentMapper mapper;
    @Autowired
    private PlatformTransactionManager manager;

    public void updateStudent() {
        Student student1 = new Student();
        student1.setId(12);
        student1.setName("hh");
        // 开启事务
        TransactionStatus status = manager.getTransaction(new DefaultTransactionDefinition());
        try {
            mapper.updateById(student1);

            // 提交事务
            manager.commit(status);
        } catch (Exception e) {
            // 回滚事务
            manager.rollback(status);
        }
    }
}

声明:部分场景参考了网上资源

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值