事务四大特性
- 原子性
- 一致性
- 隔离性
- 持久性
原子性
事务是一个不可分割的单位,要么都发生,要么都不发生。
一致性
事务执行前后数据的完整性必须保持一致。
隔离性
多个用户并发访问数据库时,一个用户的事务不能被其他用户的事务干扰。多个并发事务的数据相互隔离。
持久性
事务一旦被提交后,数据就会被持久化到数据库中。
Spring事务管理接口
- PlatformTransactionManager 事务管理器
- TransactionDefinition 事务定义信息(隔离,传播,超时,只读)
- TransactionStatus 事务具体运行状态
PlatformTransactionManager 事务管理器
Spring为不同的持久层框架提供PlatformTransactionManager接口不同的实现类。
TransactionDefinition 事务定义信息
事务定义信息接口中主要涉及事务隔离级别和事务传播行为两个内容的设置。下面就具体分别来讲。
事务隔离级别
事务隔离级别的设置是为了解决脏读,不可重复读,幻读等问题的。
下面来看下这几个问题的描述。
- 脏读
一个事务a读取了另一个事务b改写还未提交的数据,如果另一个事务b回滚,事务a读取到的数据是无效的。 - 不可重复读
在同一个事务中,多次读取同一数据读取到的结果不同。比如事务 A 多次读取同一数据,事务 B 在事务A多次读取的过程中,对数据作了更新并提交,导致事务A多次读取同一数据时,结果不一致。 - 幻读
一个事务读取了几行数据后,另一个事务插入了新的数据后,幻读就发生了。再后来的查询中,第一个事务就会发现一些原来没有的数据。比如系统管理员A将数据库中所有学生的成绩从具体分数改为ABCDE等级,但是系统管理员B就在这个时候插入了一条具体分数的记录,当系统管理员A改结束后发现还有一条记录没有改过来,就好像发生了幻觉一样,这就叫幻读。
小结:不可重复读的和幻读很容易混淆,不可重复读侧重于修改,幻读侧重于新增或删除。解决不可重复读的问题只需锁住满足条件的行,解决幻读需要锁表。
为解决上面几个问题,引入事务隔离级别的概念。
其中Spring中默认设置的default隔离级别的值就是数据库的默认隔离级别,比如mysql的默认隔离级别就是repeatable_read级别。
事务传播行为
应用架构通常分为web层,service层和持久化层。一般事务机制添加在service层上,service层中的多个方法之间的调用就涉及到事务传播机制,当事务方法被另一个事务方法调用时,必须指定事务应该如何传播。
事务传播行为有下面几种:
Spring支持两种事务管理方式
- 编程式事务管理
使用transactionTemplate类来管理事务,由于使用编程式事务管理比较繁琐,一般不予采用。 - 使用xml配置声明式事务(基于Spring AOP实现)
声明式事务涉及类TransactionProxyFactoryBean,它的注释中注明了声明式事务的配置方式。
<bean id="baseTransactionProxy" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean"
* abstract="true">
* <property name="transactionManager" ref="transactionManager"/>
* <property name="transactionAttributes">
* <props>
* <prop key="insert*">PROPAGATION_REQUIRED</prop>
* <prop key="update*">PROPAGATION_REQUIRED</prop>
* <prop key="*">PROPAGATION_REQUIRED,readOnly</prop>
* </props>
* </property>
* </bean>
在transactionAttributes属性中看到以insert和update开头的方法的方法事务传播行为是PROPAGATION_REQUIRED,即方法之间调用时,这几个方法都在同一个事务中。