1. spring 的事务属性
属性:描述物体特征的一系列值。例如:身高、体重、年龄。。。
事务属性:描述事务特征的一系列值
1.1 什么是事务属性
(1) 隔离属性
(2) 传播属性
(3) 只读属性
(4) 超时属性
(5) 异常属性
1.2 如何添加事务属性
@Transational(isolation = ,propagation = ,readOnly =,timeout = ,rollbackFor =)
1.3 事务属性详解
1.3.1 isolation(隔离属性)
概念:它描述了事务解决并发问题的特征(并发:多个事务(用户)在同一时间,访问操作了相同的数据。)
产生的问题:
(1)脏读:一个事务读取了另外一个事务中没有提交的数据。会在本事务中产生数据不一致的问题。
解决方案:@Transational(isolation =Isolation.READ_COMMITTED)让一个事务只能读取其他事务提交了的事务的数据。
(2)不可重复读:一个事务中,多次读取相同的数据,但是读取结果不一样,会在本事务中产生数据不一致的问题。(注意:1.这个不是脏读,2. 多次查询在一个事务中)
解决方案:@Transactional(isolation =Isolation.REPEATABLE_READ)
本质:一把行锁,其他事务无法操作
(3)幻影读:一个事务中,多次对整表进行查询统计,但是结果不一样,会在本事务中产生数据不一致的问题。
解决方案:@Transactional(isolation =Isolation.SERIALIZABLE)
本质:表锁
总结:
安全:SERIALIZABLE > REPEATABLE_READ > READ_COMMITTED
效率:READ_COMMITTED > REPEATABLE_READ > SERIALIZABLE
spring在设置时,要是没有指定默认的隔离属性,底层会调用数据库所设置的默认属性(mysql是REPEATABLE_READ)
1.3.2 PROPAGATION(传播属性)
概念:它描述了事务解决嵌套问题的特征
什么叫做事务嵌套:它指一个事务中,包含了若干个小的事务(也可以理解为一个service中调用了其他若干个service,而每个service中都开启了事务)。
问题:大事务中融入了很多小的事务,他们彼此影响,最终会导致外部大的事务丧失了事务的原子性。
传播属性的值 |
外部不存在事务 |
外部存在事务 |
用法 |
说明 |
REQUIRED |
开启新的事务 |
融合到外部事务中 |
@Transactional(propagation=Propagation.REQUIRED) |
适用于增删改 |
SUPPORTS |
不开启新的事务 |
融合到外部事务中 |
@Transactional(propagation=Propagation.SUPPORTS) |
适用于查询操作 |
MANDATORY |
抛出异常 |
融合到外部事务中 |
@Transactional(propagation=Propagation.MANDATORY) |
略 |
REQUIRES_NEW |
开启新的事务 |
挂起外部事务,创建新的事务 |
@Transactional(propagation=Propagation.REQUIRES_NEW) |
日志记录 |
NOT_SUPPORTED |
不开启新的事务 |
挂起外部事务 |
@Transactional(propagation=Propagation.NOT_SUPPORTED) |
略 |
NEVER |
不开启新的事务 |
抛出异常 |
@Transactional(propagation=Propagation.NEVER) |
略 |
NESTED |
执行与PROPAGATION_REQUIRED类似的操作 |
在嵌套事务内执行 |
@Transactional(propagation=Propagation.NESTED) |
略 |
默认的传播属性:@Transactional(propagation=Propagation.REQUIRED)
1.3.3 readOnly(只读属性)
针对只进行查询操作的业务方法,可以加只读属性,提高运行效率。默认:false
1.3.4 timeout(超时属性)
概念:指定了事务等待的最长时间
1. 为什么事务要进行等待:当前事务访问数据时,有可能访问的数据被别的事务进行加锁,那么此时本事务就必须进行等待。
2. 等待的时间单位:秒
3. 应用:@Transactional(timeout = 2)
4. 超时属性的默认值:-1(最终由对应的数据库来指定)
1.3.5 rollbackFor(异常属性)
Spring事务处理过程中
默认:对于RuntimeException及其子类,采用回滚的策略。
默认:对于Exception及其子类,采用的是提交的策略。