事务:事务指的是逻辑上的一组操作,这组操作要么都成功,要么都失败。
Transaction事务的四大特性ACID:
- Atomicity原子性
事务的操作要么都成功,要么都不做,只要有一个失败,就会回滚。如A转账 100给B:A转出100,B转入100,这两步要么一起完成,要么都不做。 - Consistency一致性
事务的运行前后数据的完整性必须保持一致。A有200,B有200,一共400,A转给B100后,总金额还是400。 - Isolation隔离性
并发的事务之间不会相互影响,多个并发事务之间的数据要相互隔离(设置隔离级别)。如A转100给B:若事务未完成,B的账户是不会多100的。 - Durability持久性
事务一旦提交,它所做的修改将永久的保存在数据库中,不会受其他方面的影响。
Spring中的事务管理介绍(高层抽象包括3个接口):
- PlatformTransactionManager事务管理器
- TransactionDefinition事务定义信息(隔离、传播、超时、只读)
- TransactionStatus事务具体运行状态
如果不考虑隔离性,会引发安全问题:
- 脏读:一个事务可以读取另一个尚未提交事务的修改数据。
(一个事务读取了另一个事务改写但还未提交的数据,如果这些数据被回滚,则读到的数据是无效的。) - 不可重复读:在同一个事务中,同一个查询在T1时间读取某一行,在T2时间重新读取这一行时候,这一行的数据已经发生修改,可能被更新了(update),也可能被删除了(delete)。
(在同一事务中,多次读取同一数据返回的结果有所不同。) - 幻读:在同一事务中,同一查询多次进行时候,由于其他插入操作(insert)的事务提交,导致每次返回不同的结果集。
(一个事务读取了几行记录后,另一个事务插入了几行数据,幻读就发生了。再后来的查询中,第一个事务就会发现有些原来没有的记录。)
隔离级别:
- DEFAULT:使用后端数据库默认的隔离级别(Spring中的选择项)
- READ_UNCOMMITED:允许你读取还未提交的改变了的数据。可能导致脏、幻、不可重复读。
- READ_COMMITTED:允许在并发事务已经提交后读取。可防止脏读,但幻读和不可重复读仍可发生。(Oracle默认隔离级别)
- REPEATABLE_READ:对相同字段的多次读取是一致的,除非数据被事务本身改变。可防止脏、不可重复读,但幻读仍可能发生。(MySQL默认隔离级别)
- SERIALIZABLE:完全服从ACID的隔离级别,确保不发生脏、幻、不可重复读。在所有隔离级别中是最慢的,它是典型的通过完全锁定在事务中涉及的数据表来完成的。
事务的传播行为:
解决业务层之间的相互调用,同时调用业务层中不同类的方法
- PROPAGATION_REQUIRED:支持当前事务,如果不存在,就新建一个。
- PROPAGATION_SUPPORTS:支持当前事务,如果不存在,就不使用事务。
- PROPAGATION_MANDATORY:支持当前事务,如果不存在,抛出异常。
- PROPAGATION_REQUIRES_NEW:如果有事务存在,挂起当前事务,创建一个新的事务。
- PROPAGATION_NOT_SUPPORTED:以非事务方式运行,如果有事务存在,挂起当前事务。
- PROPAGATION_NEVER:以非事务方式运行,如果有事务存在,抛出异常。
- PROPAGATION_NESTED:如果当前事务存在,则嵌套事务执行。
Spring的事务管理
- 编程式的事务管理
在实际应用中很少使用,通过TransactionTemplate手动管理事务 - 使用XML配置声明式事务
开发中推荐使用(代码侵入性最小),Spring的声明式事务是通过AOP实现的
编程式事务(不常用):
Spring提供了TransactionTemplate模板,它依赖DataSourceTransactionManager,
DataSourceTransactionManager又依赖于DataSource。
通过TransactionTemplate模板中的 execute(TransactionCallback arg) 方法执行事务的具体操作,需要传入TransactionCallback接口的实现类,或通过实现匿名内部类:
TransactionCallbackWithoutResult
在它的doInTransactionWithoutResult(TranscationStatus transactionStatus) 方法内编写事务操作。
声明式事务:
- 基于TransactionProxyFactory的代理方式
<bean class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
<!-- 配置目标对象 -->
<property name="target" ref="accountService">
<!-- 注入事务管理器 -->
<property name="transactionManager" ref="transactionManager"/>
<!-- 注入事务属性 -->
<property name="transactionAttributes" >
<props>
<!-- prop的格式:
* PROPAGATION:事务的传播行为
* ISOLATION :事务的隔离级别
* readOnly :只读
* -Exception :发生哪些异常回滚事务
* +Exception :发生哪些异常事务不回滚
-->
<prop key="transfer *">PROPAGATION_REQUIRED</prop>
</props>
</property>