隔离级别

读已提交

MatrixOne 默认读已提交(Read Committed)隔离级别,它的特点如下:

  • 在不同的事务之间,只能读到其他事务已经提交的数据,对于未提交的数据,无法查看。
  • 读已提交的隔离级别,能够有效防止脏写和脏读,但是不能避免不可重复读与幻读。

读已提交原理

  • 当一个事务开始时,数据库会为该事务生成一个独一无二的事务 ID。
  • 在生成该事务 ID 的时间戳,在每次对数据进行增、删、改、查时,TAE 自动检测对应表中是否有已被更新的时间戳,如果有,则更新时间戳为最新。
  • 在对数据操作时,TAE 将操作的数据缓存在内存中,提交事务时,TAE 将内存中的数据写入到磁盘内(数据存储到 S3 路径,或本地磁盘路径)。

读已提交示例

你可以参照下面的示例,来理解读已提交隔离级别。

首先在 MatrixOne 中,我们建立一个命名为 test 的数据库与表 t1,并插入数据:

  1. create database test;
  2. use test;
  3. CREATE TABLE t1
  4. (
  5. tid INT NOT NULL primary key,
  6. tname VARCHAR(50) NOT NULL
  7. );
  8. INSERT INTO t1 VALUES(1,'version1');
  9. INSERT INTO t1 VALUES(2,'version2');

在会话 1 中,开启一个事务:

  1. use test;
  2. begin;
  3. UPDATE t1 SET tname='version3' WHERE tid=2;
  4. SELECT * FROM t1;

在会话 1 中,我们可以看到的结果是如下:

  1. +------+----------+
  2. | tid | tname |
  3. +------+----------+
  4. | 2 | version3 |
  5. | 1 | version1 |
  6. +------+----------+

此时开启会话 2,开启一个新事务去查询 t1 的内容:

  1. use test;
  2. begin;
  3. SELECT * FROM t1;

看到的结果仍然是原始数据:

  1. +------+----------+
  2. | tid | tname |
  3. +------+----------+
  4. | 1 | version1 |
  5. | 2 | version2 |
  6. +------+----------+

在会话 2 中,修改 tid=1 的行:

  1. UPDATE t1 SET tname='version0' WHERE tid=1;

此时,在会话 1 中查询 t1 的内容仍然还是修改后的数据:

  1. SELECT * FROM t1;
  2. +------+----------+
  3. | tid | tname |
  4. +------+----------+
  5. | 1 | version1 |
  6. | 2 | version3 |
  7. +------+----------+

在会话 2 提交自己的数据后,再查询会话 1,会发现,此时会话 1 的内容已经变成了会话 2 提交之后的数据:

  • 会话 2:
  1. -- 在会话 2 中提交数据:
  2. COMMIT;
  • 会话 1:
  1. -- 查询会话 1 的内容是否已经变成了会话 2 提交之后的数据:
  2. SELECT * FROM t1;
  3. +------+----------+
  4. | tid | tname |
  5. +------+----------+
  6. | 1 | version0 |
  7. | 2 | version3 |
  8. +------+----------+

快照隔离

在 MatrixOne 中,也支持快照隔离(Snapshot Isolation),为了与 MySQL 隔离级别保持一致,MatrixOne 快照隔离又叫做可重复读(REPEATABLE READS)。该级别的隔离实现原理如下:

快照隔离原理

  • 当一个事务开始时,数据库会为该事务生成一个事务 ID,这是一个独一无二的 ID。
  • 在生成该事务 ID 的时间戳,生成一个对应数据的快照,此时事务的所有操作都是基于该快照来执行。
  • 当事务提交完成对数据的修改后,释放事务 ID 与数据快照。

快照隔离示例

你可以参照下面的示例,来帮助理解快照隔离。

首先在 MatrixOne 中,我们建立一个数据库 test 与表 t1

  1. create database test;
  2. use test;
  3. CREATE TABLE t1
  4. (
  5. tid INT NOT NULL primary key,
  6. tname VARCHAR(50) NOT NULL
  7. );
  8. INSERT INTO t1 VALUES(1,'version1');
  9. INSERT INTO t1 VALUES(2,'version2');

在会话 1 中,开启一个事务:

  1. use test;
  2. begin;
  3. UPDATE t1 SET tname='version3' WHERE tid=2;
  4. SELECT * FROM t1;

在会话 1 中,我们可以看到的结果是如下,根据快照的数据所进行的修改结果:

  1. +------+----------+
  2. | tid | tname |
  3. +------+----------+
  4. | 2 | version3 |
  5. | 1 | version1 |
  6. +------+----------+

此时开启会话 2,去查询 t1 的内容:

  1. use test;
  2. SELECT * FROM t1;

看到的结果仍然是原始数据:

  1. +------+----------+
  2. | tid | tname |
  3. +------+----------+
  4. | 1 | version1 |
  5. | 2 | version2 |
  6. +------+----------+

在会话 1 中,我们将事务提交:

  1. COMMIT;

此时,在会话 2 中查询 t1 的内容就变成了提交后的数据:

  1. SELECT * FROM t1;
  2. +------+----------+
  3. | tid | tname |
  4. +------+----------+
  5. | 1 | version1 |
  6. | 2 | version3 |
  7. +------+----------+