我注意到 Oracle 和 PostgreSQL 中都发生了以下情况。
考虑到我们有以下数据库架构:
create table post (
id int8 not null,
title varchar(255),
version int4 not null,
primary key (id));
create table post_comment (
id int8 not null,
review varchar(255),
version int4 not null,
post_id int8,
primary key (id));
alter table post_comment
add constraint FKna4y825fdc5hw8aow65ijexm0
foreign key (post_id) references post;
具有以下数据:
insert into post (title, version, id) values ('Transactions', 0, 1);
insert into post_comment (post_id, review, version, id)
values (1, 'Post comment 1', 459, 0);
insert into post_comment (post_id, review, version, id)
values (1, 'Post comment 2', 537, 1);
insert into post_comment (post_id, review, version, id)
values (1, 'Post comment 3', 689, 2);
如果我打开两个单独的 SQL 控制台并执行以下语句:
TX1: BEGIN TRANSACTION ISOLATION LEVEL SERIALIZABLE;
TX2: BEGIN TRANSACTION ISOLATION LEVEL SERIALIZABLE;
TX1: SELECT COUNT(*) FROM post_comment where post_id = 1;
TX1: > 3
TX1: UPDATE post_comment SET version = 100 WHERE post_id = 1;
TX2: INSERT INTO post_comment (post_id, review, version, id) VALUES (1, 'Phantom', 0, 1000);
TX2: COMMIT;
TX1: SELECT COUNT(*) FROM post_comment where post_id = 1;
TX1: > 3
TX1: COMMIT;
TX3: SELECT * from post_comment;
> 0;"Post comment 0";100;1
1;"Post comment 1";100;1
2;"Post comment 2";100;1
1000;"Phantom";0;1
正如预期的那样,SERIALIZABLE
隔离级别保留了 TX1 事务开始时的快照数据,TX1 只看到 3post_comment
记录。
由于Oracle和PostgreSQL中的MVCC模型,TX2被允许插入新记录并提交。
为什么允许 TX1 提交?因为这是写入倾斜异常,所以我期望看到 TX1 会因“序列化失败异常”或类似的情况而回滚。
PostgreSQL 和 Oracle 中的 MVCC Serialized 模型是否仅提供快照隔离保证,但没有 Write Skew 异常检测?
UPDATE
我什至更改了 Tx1 以发出更新语句来更改version
所有人的专栏post_comment
属于同一记录post
.
这样,Tx2 创建一条新记录,并且 Tx1 将在不知道已添加满足 UPDATE 过滤条件的新记录的情况下提交。
实际上,使其在 PostgreSQL 上失败的唯一方法是在插入虚拟记录之前在 Tx2 中执行以下 COUNT 查询:
Tx2: SELECT COUNT(*) FROM post_comment where post_id = 1 and version = 0
TX2: INSERT INTO post_comment (post_id, review, version, id) VALUES (1, 'Phantom', 0, 1000);
TX2: COMMIT;
然后 Tx1 将被回滚:
org.postgresql.util.PSQLException: ERROR: could not serialize access due to read/write dependencies among transactions
Detail: Reason code: Canceled on identification as a pivot, during conflict out checking.
Hint: The transaction might succeed if retried.
最有可能的是写入倾斜异常预防机制检测到此更改并回滚事务。
有趣的是,Oracle 似乎并没有受到这种异常的困扰,因此 Tx1 只是成功提交。由于 Oracle 无法防止写入偏差的发生,因此 Tx1 提交得很好。
顺便说一句,您可以自己运行所有这些示例,因为它们已打开GitHub https://github.com/vladmihalcea/high-performance-java-persistence/tree/master/core/src/test/java/com/vladmihalcea/book/hpjp/jdbc/transaction.