-- ==============================================
-- 会话A - 幻读实验主事务(整段复制,按执行提示分步跑)
-- ==============================================
-- 1. 环境准备:删旧表+建测试表+关自动提交+确认隔离级别
DROP TABLE IF EXISTS test_phantom;
CREATE TABLE test_phantom (id INT PRIMARY KEY, name VARCHAR(20) NOT NULL) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
SET SESSION autocommit = 0;
SELECT @@transaction_isolation AS '当前隔离级别'; -- 预期结果:REPEATABLE-READ
-- 2. 启动事务+首次快照读(创建ReadView,核心步骤)
START TRANSACTION;
SELECT * FROM test_phantom WHERE id=1; -- 预期结果:空集(表无数据)
-- ========== 到此为止!先停在这里,切换到【会话B】执行完所有代码,再切回A继续下面的步骤 ==========
-- 3. 二次快照读(复用ReadView,依然读不到数据,RR纯快照读特性)
SELECT * FROM test_phantom WHERE id=1; -- 预期结果:还是空集(没幻读,防住了纯快照读)
-- 4. 执行UPDATE当前读(绕开ReadView,抓取会话B提交的最新数据)
UPDATE test_phantom SET name='innodb_phantom' WHERE id=1; -- 预期结果:Rows matched:1 Changed:1(执行成功)
-- 5. 三次快照读(幻读出现!同一事务相同查询,从空变有)
SELECT * FROM test_phantom WHERE id=1; -- 预期结果:(1, 'innodb_phantom') 【核心幻读现象】
-- 6. 提交事务,结束实验
COMMIT;
-- ==============================================
-- 会话B - 插入数据并提交(整段复制,直接全跑)
-- ==============================================
-- 环境准备:仅关闭自动提交即可(表由会话A创建,无需重复建)
SET SESSION autocommit = 0;
-- 插入id=1数据并立即提交(数据对外可见,触发幻读的前提)
START TRANSACTION;
INSERT INTO test_phantom VALUES (1, 'mysql_rr'); -- 预期结果:插入成功,影响行数1
COMMIT; -- 提交后数据持久化,对其他会话可见