Oracle与MySQL中“可串行化”的对比测试
Thomas Kyte (Oracle 世界中那个著名的 Tom)在“Oracle 9i&10g编程艺术”一书中为了说明事务的隔离级别,提供了一些例子。这里我们来看看“可串行化”级别下的表现。
以普通用户登录系统,并创建测试用的表。
idle> connect scott/tiger
Connected.
scott@FOO> select * from v$version;
BANNER
--------------------------------------------------------------------------------
Oracle Database 11g Enterprise Edition Release 11.2.0.1.0 - Production
PL/SQL Release 11.2.0.1.0 - Production
CORE 11.2.0.1.0 Production
TNS for Linux: Version 11.2.0.1.0 - Production
NLSRTL Version 11.2.0.1.0 - Production
scott@FOO> create table a(x int);
Table created.
scott@FOO> create table b(x int);
Table created.
修改事务的隔离级别为“可串行化”。注意,下面的代码可以左右滚动以显示完整的两个事务。
时刻 会话1 会话2
t1 scott@FOO> alter session set isolation_level = serializable;
Session altered.
t2 scott@FOO> alter session set isolation_level = serializable;
Session altered.
t3 scott@FOO> insert into a select count(*) from b;
1 row created.
t4 scott@FOO> insert into b select count(*) from a;
1 row created.
t5 scott@FOO> commit;
Commit complete.
t6 scott@FOO> commit;
Commit complete.
让我们来检查一下结果。
scott@FOO> select * from a;
X
----------
0
scott@FOO> select * from b;
X
----------
0
确实如 Tom 所说,两个表的记录值都是 0。这是因为会话1和会话2中的事务看到的都是事务开始时的数据库状态。在 Oracle 中,可串行化模式下并不能做到我们通常理解的那样:会话1和会话2对应的事务依次串行执行的结果。如果会话1先执行,会话2后执行,则结果应该是表a中的值为0;表b中的值为1。反之,a的值为1;b的值为0。Oracle 的可串行化实际上是 SNAPSHOT ISOLATION,上面描述的这个现象被称为 WRITE SKEW 。
让我们看看 MySQL 的表现。
mysql> select @@version;
+-----------+
| @@version |
+-----------+
| 5.1.45 |
+-----------+
1 row in set (0.01 sec)
MySQL 中有多个存储引擎,记得选择事务支持比较好的innodb。
mysql> create table a(x int) engine = 'innodb';
Query OK, 0 rows affected (0.04 sec)
mysql> create table b(x int) engine = 'innodb';
Query OK, 0 rows affected (0.04 sec)
MySQL中默认是一语句一提交的,和 Oracle 不一样。我们把它修改为非自动提交。下面的内容可以左右滚动。
mysql> set autocommit = 0;
t1 mysql> set session transaction isolation level serializable;
Query OK, 0 rows affected (0.00 sec)
mysql> set autocommit = 0;
t2 mysql> set session transaction isolation level serializable;
Query OK, 0 rows affected (0.00 sec)
t3 mysql> insert into a select count(*) from b;
Query OK, 1 row affected (0.01 sec)
Records: 1 Duplicates: 0 Warnings: 0
t4 mysql> insert into b select count(*) from a;
因为 MySQL 在这里也采用了封锁机制,这时会话会停在这里,直到会话1输入
commit 并执行后才会出现下面的提示:
Query OK, 1 row affected (7.51 sec)
Records: 1 Duplicates: 0 Warnings: 0
t5 mysql> commit;
Query OK, 0 rows affected (0.00 sec)
t6 mysql> commit;
Query OK, 0 rows affected (0.00 sec)
让我们来查查最后的数据:
mysql> select * from a;
+------+
| x |
+------+
| 0 |
+------+
1 row in set (0.01 sec)
mysql> select * from b;
+------+
| x |
+------+
| 1 |
+------+
1 row in set (0.00 sec)
这相当于会话1中的事务先执行,会话2中的事务后执行。正好符合我们对“可串行化”的理解。如果把 MySQL 的隔离级别下降为 Repeatable Read,会发现它仍然不会有这个例子的 Write Skew 的问题。 因为针对 insert 语句中的读,MySQL 采用的仍然是当前读(而不是快照读),并且对相关的记录采取了行锁和 Gap 锁。在 RR 的隔离级别下,它甚至也不存在幻读的问题。那么 MySQL 的可串行化和 RR 的区别在哪里呢?在可串行化级别下,MySQL 退化为封锁模式,所有的读操作都是当前读,不再支持快照读,按照封锁协议加锁。可以发现,这时 MySQL 存在和封锁系统一样的缺点,读写冲突将造成较差的并发事务性能。
在说明 Oracle 有优势的特点时,Tom 总忘不了去强调一下它有多好;在说明 Oracle 不那么有优势,甚至不一定合理的地方时,他会说“这个概念经常被误解”。其实,这些技术细节上的差异都是多版本并发控制和封锁并发控制导致的;而 MySQL/InnoDB 又比较特别,综合了多版本并发控制和封锁并发控制机制。所以说,凡事有利必有弊,我们需要根据不同需要去取舍。 References:
- Oracle 9i&10g 编程艺术
- http://smalldatum.blogspot.com/2019/09/innodb-repeatable-read-in-nutshell.html, see also the comments by Marko.