什么是事务
事务是逻辑上的一组操作,要么都执行,要么都不执行。
事务能否生效数据库引擎是否支持事务是关键。比如常用的 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
面向切面的,它将具体业务与事务处理部分解耦,代码侵入性很低,所以在实际开发中声明式事务用的比较多。声明式事务也有两种实现方式,一是基于TX
和AOP
的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注解的属性
-
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
。
-
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的隔离级别,所有事务依次执行,这样事务之间就不会产生干扰,但是会严重影响程序性能。
-
timeout
事务超时时间timeout
:事务的超时时间,默认值为 -1。如果超过该时间限制但事务还没有完成,则自动回滚事务。
-
readOnly
属性readOnly
:指定事务是否为只读事务,默认值为 false;为了忽略那些不需要事务的方法,比如读取数据,可以设置 read-only 为 true。- 如果你一次执行多条查询语句,例如统计查询,报表查询,在这种场景下,多条查询 SQL 必须保证整体的读一致性,否则,在前条 SQL 查询之后,后条 SQL 查询之前,数据被其他用户改变,则该次整体的统计查询将会出现读数据不一致的状态,此时,应该启用事务支持
-
rollbackFor
属性rollbackFor
:用于指定能够触发事务回滚的异常类型,可以指定多个异常类型。
-
noRollbackFor
属性noRollbackFor
:抛出指定的异常类型,不回滚事务,也可以指定多个异常类型。
事务失效场景
java开发经常使用到@Transactional
注解控制事务,比较方便,但是有时候莫名的失效了,下面说下一些失效的场景:
1. @Transactional
应用在非 public
修饰的方法上
如果Transactional
注解应用在非public
修饰的方法上,Transactional将会失效
之所以会失效是因为在Spring AOP
代理时, TransactionInterceptor
(事务拦截器)在目标方法执行前后进行拦截,DynamicAdvisedInterceptor
(CglibAopProxy 的内部类)的 intercept 方法或 JdkDynamicAopProxy
的 invoke 方法会间接调用 AbstractFallbackTransactionAttributeSource
的 computeTransactionAttribute
方法,获取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);
}
}
}
声明:部分场景参考了网上资源