Skip to content

4.05 关系数据库的事务处理

关系数据管理系统提供了事务处理功能用于确保应用层业务流程的原子性。事务是关系数据库事务处理的最小单元,也是应用程序的基本逻辑单元,它由一系列的数据库增删改查操作组成,用于描述应用层的业务逻辑。本节主要介绍关系数据库的事务基本特性。

4.05.1 事务的定义

在关系数据库中,事务的定义包含以下三个关键字:

BEGIN TRANSACTION /*事务开始*/
COMMIT /*事务提交*/
ABORT /*事务回滚*/
事务由BEGIN TRANSACTION和COMMIT/ABORT之间的所有SQL语句组成。其中,BEGIN TRANSACTION表示事务的开始,COMMIT和ABORT表示事务的结束。COMMIT表示事务提交,即事务中所有UPDATE、DELETE、INSERT操作产生的结果已写回磁盘,事务正常结束;ABORT表示事务回滚,即在执行过程中事务因某种故障不能继续执行,那么撤销已经完成的操作,使数据库回滚到事务开始之前的状态。

下面给出转账交易在关系数据管理系统中的事务定义。

账户U1向账户U2转账50元(事务T0)。
BEGIN TRANSACTION
SELECT Balance A FROM Deposit WHERE ID='U1' FOR UPDATE;
IF(A < 50) 
THEN ROLLBACK; 
ELSE
{
UPDATE Deposit SET  Balance = A - 50 WHERE ID='U1';
UPDATE Deposit SET  Balance = Balance + 50 WHERE ID='U2';
COMMIT;
}

4.05.2 事务的ACID性质

关系数据库的事务通常被赋予了四个特性,分别为:原子性(Atomicity)、一致性(Consistency)、隔离性(Isolation)和持久性(Durability)。这四个特性简称事务的ACID特性。

  • 原子性:事务中的所有操作要么都执行,要么都不执行,不会出现中间状态。该性质通常由日志机制来保障。
  • 一致性:事务执行后必须保证数据库状态必须满足事先定义的一致性约束。比如,银行转账业务可以事先定义如下约束:事务执行前和执行后所有账户的总额度不发生变化,并且任何账户的余额都不能为负数。为了实现事务的一致性,既需要依赖系统的日志和并发控制机制,常常也需要应用程序逻辑正确性的配合。
  • 隔离性:多个并发执行的事务之间相互干扰,彼此之间是隔离的。由于不同应用对事务相互干扰的容忍度不同,关系数据库通常允许应用开发人员对事务的隔离级别进行设置。通常,关系数据库系统提供的隔离级别包括读未提交(Read Uncommitted)、读已提交(Read Committed)、可重复读(Repetable Read)、可串行化(Serializable)、快照读(Snapshot)等。该性质通常由并发控制机制来保障。
  • 持久性:一个事务一旦提交之后,它对数据库产生的数据修改是永久性的并且不能撤销。之后的其他操作或者故障都不会对执行结果产生任何影响。该性质通常由日志机制来保障。

练习题

1. 关系数据库系统使用了undo/redo日志,在故障发生后,发现日志记录如下:

<T1,START>
<T1,A,100,200>   /*事务T1将数据项A从100修改为200*/
<T2,START>
<T2,B,50,150>
<T1,COMMIT>
<CHECKPOINT>
<T3,START>
<T3,C,300,400>
<T3,COMMIT>

请问:系统恢复后A,B,C的取值是多少?

  1. A=100, B=150, C=400
  2. A=200, B=50, C=400
  3. A=100, B=50, C=300
  4. A=200, B=150, C=300

2. 当前有两个事务T1,T2。两个事务的操作如下:

T1: R(A), W(A), R(B), W(B)  /*事务T1读数据项A,再将更新后的值写回数据项A,T1读数据项B,再将更新后的值写回数据项B*/
T2: R(A), W(A), R(B), W(B)  /*事务T2读数据项A,再将更新后的值写回数据项A,T2读数据项B,再将更新后的值写回数据项B*/

以下哪种两个事务的调度顺序保证了事务的正确性?用T1:R(A)表示事务T1读A,T1:W(A)表示事务T1写A。

  1. T1:R(A); T2:R(A); T1:W(A); T2:W(A); T1:R(B); T2:R(B); T1:W(B); T2:W(B)
  2. T1:R(A); T1:W(A); T2:R(A); T2:W(A); T2:R(B); T2:W(B); T1:R(B); T1:W(B)
  3. T1:R(A); T1:W(A); T2:R(A); T2:W(A); T1:R(B); T1:W(B); T2:R(B); T2:W(B)
  4. 以上都不对

3. 事务T1和事务T2并发执行,T1修改数据A后提交,T2读取数据项A两次(分别在T1提交前和提交后)。请思考:

  • 若关系数据库隔离级别为READ COMMITTED,T2的两次读取结果是否可能不同?为什么?
  • 若隔离级别为REPEATABLE READ,结果是否可能不同?为什么?

上一页<< | >>下一页