事务
事务是数据库中的一系列操作,这些操作要么全部完成,要么全部不完成,事务必须遵循ACID规则。
ACID
ACID是数据库事务正确执行的四个基本要素的缩写。包括:
- Atomicity(原子性) :事务中的一系列操作,要么全部完成,要么全部不完成;
- Consistency(一致性) :事务开始前和结束后,数据库的完整性么有被破坏,但执行过程中可以暂时破坏;
- Isolation(隔离性) :事务与事务之间是相互独立的,互补干扰,一个事务不知道另一事务执行带来的数据变更,也无法与另一事务交互;
- Durability(持久性) :事务完成以后,事务对数据库产生的变化持久保存在数据库中。
事务的隔离级别
对于同时运行的多个事务,这些事务访问数据库中的相同数据,如果没有采用隔离机制就会导致各种并发问题:
- 脏读:对于两个事务T1、T2,如果T1读取了被T2更新但还没有commit的数据之后,T2发生回滚,那么T1读取的数据就是临时的、无效的。
- 不可重复读:对于两个事务T1、T2,读取了同一数据,然后T2更新了该数据,T1再读取到的该数据就不一样了。
- 幻读:对于两个事务T1、T2,T1读取了一个表,T2在该表中插入了一些新的记录,T1再读取该表的时候就发现这些新的行。
数据库系统必须具有隔离并发运行的各个事务的能力,使彼此之间互不影响,避免发生并发问题。
一个事务与其它事务的隔离程度,成为事务的隔离级别。隔离级别越高,数据一致性就越好,但并发性能就越弱。
MySQL提供了4种隔离级别:
-
READ UNCOMMITTED(读未提交数据)
允许读取尚未提交的修改,可能导致脏读、幻读和不可重复读 -
READ COMMITTED(读已提交数据)
允许从已经提交的事务读取,可防止脏读,但幻读、不可重复读仍然有可能发生 -
REPEATABLE READ(可重复读) [默认的]
对相同字段的多次读取的结果是一致的,除非数据被当前事务自身修改。可防止脏读和不可重复读,但幻读仍有可能发生。(对应数据库的行锁
) -
SERIALIZABLE(串行化)
完全服从ACID隔离原则,确保不发生脏读、不可重复读、和幻读,但执行效率最低。(对应数据库的表锁
)
mysql上查看当前事务隔离级别:select @@tx_isolation
mysql> select @@tx_isolation;
+-----------------+
| @@tx_isolation |
+-----------------+
| REPEATABLE-READ |
+-----------------+
1 row in set (0.00 sec)
修改全局事务隔离级别:
set global transaction isolation level read committed;
Spring声明式事务
Spring提供了对事务的支持,可以通过简单配置和注解的方式快速无侵入地实现事务。
添加事务依赖
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-tx</artifactId>
<version>${spring.version}</version>
</dependency>
Spring配置文件中配置事务管理器、开启注解
<!-- 配置事务管理器 -->
<bean name="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
</bean>
<!--配置基于注解的声明式事务-->
<tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true"/>
@Transactionl注解可以用于方法和类(包括接口)上,建议仅在需要的方法上使用此注解
@Transactional(propagation = Propagation.REQUIRED,isolation = Isolation.DEFAULT)
更详细的说明见我的另一篇文章:Spring声明式事务