事务隔离
此条目需要精通或熟悉相关主题的编者参与及协助编辑。 (2017年2月26日) |
此条目需要补充更多来源。 (2009年1月) |
事务隔离(英语:Transaction Isolation)定义了数据库系统中一个事务中操作的结果在何时以何种方式对其他并发事务操作可见。隔离是事务ACID(原子性、一致性、隔离性、持久性)四大属性之一。
并发控制
[编辑]并发控制描述了数据库事务隔离以保证数据正确性的机制。为了保证并行事务执行的准确执行,数据库和存储引擎在设计的时候着重强调了并发控制这一点。典型的事务相关机制限制数据的访问顺序(执行调度)以满足可序列化和可恢复性。限制数据访问意味着降低了执行的性能,并发控制机制就是要保证在满足这些限制的前提下提供尽可能高的性能。在不损害正确性的情况下,可序列化的要求经常会为了性能而妥协,但是为了避免数据一致性的破坏,可恢复性不能够妥协。
两阶段锁是关系数据库中最常见的提供了可序列化和可恢复性的并发控制机制,为了访问一个数据库对象,事务首先要获得这个对象的锁。对于不同的访问类型(如对对象的读或写操作)和锁的类型,如果另外一个事务正持有这个对象的锁,获得锁的过程会被阻塞或者延迟。
读现象举例
[编辑]ANSI/ISO SQL 92标准描述了三种不同的一个事务读取另外一个事务可能修改的数据的“读现象”。
下面的例子中,我们假设有两个事务,事务1执行语句1。接着,事务2执行语句2并且提交,最后事务1再执行语句1。
查询使用如下的数据表。
id | name | age |
---|---|---|
1 | Joe | 20 |
2 | Jill | 25 |
脏读
[编辑]当一个事务允许读取另外一个事务修改但未提交的数据时,就可能发生脏读(dirty reads)。
脏读和不可重复读类似,不同点在于事务2不需要提交就能造成语句1两次执行的结果不同。在未提交读隔离级别唯一禁止的是更新混乱,即早期的更新可能出现在后来更新之前的结果集中。
在我们的例子中,事务2修改了一行,但是没有提交,事务1读了这个没有提交的数据。现在如果事务2回滚了刚才的修改或者做了另外的修改的话,事务1中查到的数据就是不正确的了。
事务 1 | 事务 2 |
---|---|
/* Query 1 */
SELECT age FROM users WHERE id = 1;
/* will read 20 */
|
|
/* Query 2 */
UPDATE users SET age = 21 WHERE id = 1;
/* No commit here */
| |
/* Query 1 */
SELECT age FROM users WHERE id = 1;
/* will read 21 */
|
|
ROLLBACK; /* lock-based DIRTY READ */
|
在这个例子中,事务2回滚后就没有id是1,age是21的数据行了。
不可重复读
[编辑]在一次事务中,当一行数据获取两遍得到不同的结果表示发生了不可重复读(non-repeatable reads).
在基于锁的并发控制中“不可重复读”现象发生在当执行SELECT操作时没有获得读锁或者SELECT操作执行完后马上释放了读锁; 多版本并发控制中当没有要求一个提交冲突(commit conflict)的事务回滚也会发生“不可重复读”现象。
事务 1 | 事务 2 |
---|---|
/* Query 1 */
SELECT * FROM users WHERE id = 1;
|
|
/* Query 2 */
UPDATE users SET age = 21 WHERE id = 1;
COMMIT; /* in multiversion concurrency
control, or lock-based READ COMMITTED */
| |
/* Query 1 */
SELECT * FROM users WHERE id = 1;
COMMIT; /* lock-based REPEATABLE READ */
|
在这个例子中,事务2提交成功,因此他对id为1的行的修改就对其他事务可见了。但是事务1在此前已经从这行读到了另外一个“age”的值。在可序列化(SERIALIZABLE)和可重复读的隔离级别中,数据库在第二次SELECT请求的时必须返回更新之前的值。在提交读和未提交读中,返回的是更新之后的值,这个现象就是不可重复读。
有两种策略可以避免不可重复读。一个是要求事务2延迟到事务1提交或者回滚之后再执行。这种方式实现了T1, T2 的串行化调度。串行化调度可以支持可重复读。
另一种被用在多版本并发控制的策略是允许事务2先提交,这样能得到更好的并发性能。但因为事务1在事务2之前开始,事务1必须在其开始执行时间点的数据库的快照上面操作。当事务1最终提交时候,数据库会检查其结果是否等价于T1, T2串行调度。如果等价,则允许事务1提交,如果不等价,事务1需要回滚并抛出个串行化失败的错误。
使用基于锁的并发控制,在可重复读的隔离级别中,ID=1的行会被锁住,在事务1提交或回滚前一直阻塞语句2的执行。在提交读的级别,语句1第二次执行,age已经被修改了。
在多版本并发控制机制下,可序列化(SERIALIZABLE)级别,两次SELECT语句读到的数据都是事务1开始的快照,因此返回同样的数据。但是,如果事务1试图UPDATE这行数据,事务1会被要求回滚并抛出一个串行化失败的错误。
在提交读隔离级别,每个语句读到的是语句执行前的快照,因此读到更新前后不同的值。在这种级别不会有串行化的错误(因为这种级别不要求串行化),事务1也不要求重试。
幻影读
[编辑]在事务执行过程中,当两个完全相同的查询语句执行得到不同的结果集。这种现象称为“幻影读(phantom read)”
当事务没有获取范围锁的情况下执行SELECT ... WHERE操作可能会发生“幻影读”。
“幻影读”是不可重复读的一种特殊场景:当事务1两次执行SELECT ... WHERE检索一定范围内数据的操作中间,事务2在这个表中创建了(如INSERT)了一行新数据,这条新数据正好满足事务1的“WHERE”子句。
事务 1 | 事务 2 |
---|---|
/* Query 1 */
SELECT * FROM users
WHERE age BETWEEN 10 AND 30;
|
|
/* Query 2 */
INSERT INTO users VALUES ( 3, 'Bob', 27 );
COMMIT;
| |
/* Query 1 */
SELECT * FROM users
WHERE age BETWEEN 10 AND 30;
|
需要指出的是事务1执行了两遍同样的查询语句。如果设了最高的隔离级别,两次会得到同样的结果集,这也正是数据库在可序列化(SERIALIZABLE)隔离级别上需要满足的。但是在较低的隔离级别上,第二次查询可能会得到不同的结果集。
在可序列化隔离级别,查询语句1在age从10到30的记录上加锁,事务2只能阻塞直至事务1提交。在可重复读级别,这个范围不会被锁定,允许记录插入,因此第二次执行语句1的结果中会包括新插入的行。
隔离级别
[编辑]在数据库事务的ACID四个属性中,隔离性是一个限制最宽松的。为了获取更高的隔离等级,数据库系统的通常使用锁机制或者多版本并发控制机制。 应用软件也需要额外的逻辑来使其正常工作。
很多数据库管理系统(DBMS)定义了不同的“事务隔离等级”来控制锁的程度。在很多数据库系统中,多数的事务都避免高等级的隔离等级(如可串行化)从而减少锁的开销。程序员需要小心的分析数据库访问部分的代码来保证隔离级别的降低不会造成难以发现的代码bug。相反的,更高的隔离级别会增加死锁发生的几率,同样需要编程过程中去避免。
由于更高的隔离级别中不存在被一个更低的隔离级别禁止的操作,DBMS被允许使用一个比请求的隔离级别更高的隔离级别。
可串行化
[编辑]可串行化(SERIALIZABLE)是最高的隔离级别。
在基于锁机制并发控制的DBMS上,可串行化要求在选定对象上的读锁和写锁直到事务结束后才能释放。在SELECT的查询中使用一个“WHERE”子句来描述一个范围时应该获得一个“范围锁”(range-locks)。这种机制可以避免“幻影读”现象。
当采用不基于锁的并发控制时不用获取锁。但当系统探测到几个并发事务有“写冲突”的时候,只有其中一个是允许提交的。这种机制的详细描述见快照隔离。
可重复读
[编辑]在可重复读(REPEATABLE READS)隔离级别中,基于锁机制并发控制的DBMS需要对选定对象的读锁(read locks)和写锁(write locks)一直保持到事务结束,但不要求“范围锁”,因此可能会发生“幻影读”。
提交读
[编辑]在提交读(READ COMMITTED)级别中,基于锁机制并发控制的DBMS需要对选定对象的写锁一直保持到事务结束,但是读锁在SELECT操作完成后马上释放(因此“不可重复读”现象可能会发生,见下面描述)。和前一种隔离级别一样,也不要求“范围锁”。
未提交读
[编辑]未提交读(READ UNCOMMITTED)是最低的隔离级别。允许“脏读”(dirty reads),事务可以看到其他事务“尚未提交”的修改。
通过比低一级的隔离级别要求更多的限制,高一级的级别提供更强的隔离性。标准允许事务运行在更强的事务隔离级别上。(如在可重复读隔离级别上执行提交读的事务是没有问题的)
默认隔离级别
[编辑]不同的DBMS默认隔离级别也不同。大多数据库允许用户设置隔离级别。有些DBMS在执行一个SELECT语句时使用额外的语法来获取锁(如SELECT ... FOR UPDATE来获得在访问的数据行上的排他锁)。
隔离级别、读现象和锁
[编辑]隔离级别vs读现象
[编辑]隔离级别 | 脏读 | 不可重复读 | 幻影读 |
---|---|---|---|
未提交读 | 可能发生 | 可能发生 | 可能发生 |
提交读 | - | 可能发生 | 可能发生 |
可重复读 | - | - | 可能发生 |
可序列化 | - | - | - |
“可能发生”表示这个隔离级别会发生对应的现象,“-”表示不会发生。
值得一提的是,避免以上三种现象虽然可以满足 ANSI 对可串行化(Serializable)级别的定义,但是其并非真正的可串行化,不能保证执行效果和串行执行完全一致,可能会出现串行化异常,例如写偏差和只读事务偏差。
隔离级别vs 锁持续时间
[编辑]在基于锁的并发控制中,隔离级别决定了锁的持有时间。"C"-表示锁会持续到事务提交。 "S" –表示锁持续到当前语句执行完毕。如果锁在语句执行完毕就释放则另外一个事务就可以在这个事务提交前修改锁定的数据,从而造成混乱。
隔离级别 | 写操作 | 读操作 | 范围操作 (...where...) |
---|---|---|---|
未提交读 | S | S | S |
提交读 | C | S | S |
可重复读 | C | C | S |
可序列化 | C | C | C |
参考文献
[编辑]相关条目
[编辑]外部链接
[编辑]- Oracle® Database Concepts(页面存档备份,存于互联网档案馆), chapter 13 Data Concurrency and Consistency, Preventable Phenomena and Transaction Isolation Levels(页面存档备份,存于互联网档案馆)
- Oracle® Database SQL Reference(页面存档备份,存于互联网档案馆), chapter 19 SQL Statements: SAVEPOINT to UPDATE(页面存档备份,存于互联网档案馆), SET TRANSACTION(页面存档备份,存于互联网档案馆)
- in JDBC: Connection constant fields(页面存档备份,存于互联网档案馆), Connection.getTransactionIsolation()(页面存档备份,存于互联网档案馆), Connection.setTransactionIsolation(int)(页面存档备份,存于互联网档案馆)
- in Spring Framework: @Transactional, Isolation