1.什么是事务(Transaction)
一组业务逻辑(如:ABCDEF)要么全部执行成功,要么全部执行不成功的操作就叫做事务。
1.1事务的特性:
原子性(Atomicity):一个事务就如同一个原子,一个不可分割的部分。
一致性(Consistency):如同执行数据库操作后,数据不会破坏。(假设给A账户转钱,A账户扣了钱,不可能B账户没有增加钱)
隔离性(Transaction isolation Level):关于事务的并发。(假设我们编写了一条update语句对数据库进行操作,同时别 人也编写了一条delete对相同的数据进行操作,可以想象,如果先执行了delete后,再执行update就会报错)
持久性(Durability):当我们对数据库执行了insert操作后,数据库就应该保证有一条数据永久性的存放在磁盘中。
1.2隔离性问题:
脏读(Dirty Read):一个事务读到了另一个事务没有提交的数据。(例如:在T1时刻事务A开始事务。在T2时刻事务A查询账户余额为1000,同时在T2时刻事务B开始事务。在T3时刻事务A从账户取出100元并把余额改为900。在T4时刻,事务B查询账户余额为900。在T5时刻事务A撤销事务,并把余额恢复为1000。在T6时刻事务B向账户存入100,并把余额改为1000。在T7时刻事务B提交事务。然而此时账户余额应该为1100,而实际上账户余额只有1000,这样就违背了事务的一致性)
不可重复读(Unrepeatable Read):一个事务读取到了另一个事务已经提交了的数据(update)。(例如:在T1时刻事务A开始事务。在T2时刻事务B开始事务。在T3时刻事务B查询账户余额为1000。在T4时刻事务A查询账户余额为1000。在T5时刻事务B从账户中存入100。在T6时刻事务B提交事务。在T7时刻事务A从账户取出100。在T8时刻事务A撤销事务将余额恢复为1000。然而事实上事务B向账户存入了100,账户余额为1100。而当事务A回滚的时候将账户余额改为了1000)
幻读(Phantom Read):一个事务读取到了另一个事务已经提交了的数据(insert)。(例如:在T1时刻事务A开始事务,在T2时刻事务B开始事务。在T3时刻事务A统计总存款为1000。在T4时刻事务B取出100,并把余额改为900。在T5时刻事务B提交事务。在T6时刻事务A向账号中汇入100并把余额改为1100。在T7时刻事务A提交事务。然而实际上事务B已经取出了100,事务A存入100是在事务B取出100前进行读取余额为1000,这样就导致了数据增多)
1.3隔离级别:
Read Committed(读已提交):解决了脏读问题,而没有解决不可重复读和幻读。
Read Uncommitted(读未提交):存在3个问题。
Repeatable Read(可重复读):解决脏读和不可重复读。存在一个问题。
Serializable(串行化):解决了所有问题。
JDBC也提供了这四种事务的隔离级别。四种隔离级别的执行效率按(Read Uncommitted》Read Committed》Repeatable Read》Serializable)递减。
2.Spring对事务的管理介绍
2.1需要导入jar包(Spring-tx.jar)
2.2spring-tx.jar包中的三个顶级接口介绍:
PlatformTransactionManager(平台事务管理器):要通过Spring管理事务就必须使用事务管理器,因此就必须配置事务管理器。
由于Spring-tx.jar包中就只定义了平台事务管理器的接口,并没实现事务管理,因此需要提供实现类,因此就需要导入以下jar包(spring-jdbc.jar这是Spring整合jdbc开发用的。spring-orm.jar这是Spring整合hibernate开发用的)。常见的事管理器有:DataSourceTransactionManager(jdbc开发时事务管理器)、HibernateTransactionManager(hibernate开发时事务管理器)
TransactionDefinition(事务详情、事务定义、事务属性):Spring用于确定事务的具体详情,例如:隔离级别、是否只读、超时时间等。
Spring在事务详情中提供了事务的传播行为(两个业务之间如何共享事务):
PROPAGATION_REQUIRED , required , 必须 【默认值】
支持当前事务,A如果有事务,B将使用该事务。
如果A没有事务,B将创建一个新的事务。
PROPAGATION_SUPPORTS ,supports ,支持
支持当前事务,A如果有事务,B将使用该事务。
如果A没有事务,B将以非事务执行。
PROPAGATION_MANDATORY,mandatory ,强制
支持当前事务,A如果有事务,B将使用该事务。
如果A没有事务,B将抛异常。
PROPAGATION_REQUIRES_NEW