PostgreSQL 事务隔离级别详解(以及与MySQL实现差异)

一、什么是事务隔离级别

在数据库中,**事务隔离级别(Transaction Isolation Level)**用于控制多个事务并发执行时的可见性规则。

SQL 标准定义了四种隔离级别:

隔离级别 可能出现的问题
Read Uncommitted 脏读、不可重复读、幻读
Read Committed 不可重复读、幻读
Repeatable Read 幻读
Serializable 无并发问题

常见的并发问题:

1️⃣ 脏读(Dirty Read)

读取到另一个事务 未提交的数据

2️⃣ 不可重复读(Non-repeatable Read)

同一事务内两次读取同一行数据结果不同

3️⃣ 幻读(Phantom Read)

同一事务两次查询 返回的行数不同


二、PostgreSQL支持的隔离级别

PostgreSQL同样支持SQL标准的四种隔离级别,但实际实现略有不同

隔离级别 PG支持情况 说明
Read Uncommitted 实际等同于RC PG不会出现脏读
Read Committed 默认级别 每条SQL看到已提交数据
Repeatable Read 完整支持 使用Snapshot Isolation
Serializable 完整支持 使用SSI算法

三、Read Committed(默认隔离级别)

PostgreSQL 默认隔离级别是:

复制代码
READ COMMITTED

特点:

  • 每条 SQL 都读取 当前已提交数据

  • 不会脏读

  • 可能出现不可重复读

  • 可能出现幻读

示例

事务A:

复制代码
BEGIN;

SELECT balance FROM account WHERE id = 1;

事务B:

复制代码
UPDATE account SET balance = 200 WHERE id = 1;
COMMIT;

事务A再次查询:

复制代码
SELECT balance FROM account WHERE id = 1;

结果可能变化,这就是 不可重复读


四、Repeatable Read(可重复读)

在 PostgreSQL 中:

复制代码
REPEATABLE READ

是基于 Snapshot Isolation(快照隔离) 实现的。

事务开始时会创建一个 数据快照(Snapshot)

复制代码
事务开始 → 创建 snapshot
之后所有查询都基于这个 snapshot

特点:

  • 不脏读

  • 不可重复读

  • 不会出现幻读

  • 不需要锁

示例:

复制代码
BEGIN;
SET TRANSACTION ISOLATION LEVEL REPEATABLE READ;

SELECT * FROM orders WHERE price > 100;

即使其他事务插入新数据:

复制代码
INSERT INTO orders VALUES (...)

当前事务也 看不到新增数据

这就是 快照隔离


五、Serializable(可串行化)

Serializable 是 最高隔离级别

PostgreSQL 使用:

复制代码
SSI(Serializable Snapshot Isolation)

核心思想:

  • 不加锁

  • 通过检测冲突保证串行执行效果

如果检测到冲突:

复制代码
ERROR: could not serialize access due to concurrent update

此时需要 应用层重试事务

示例:

复制代码
BEGIN;
SET TRANSACTION ISOLATION LEVEL SERIALIZABLE;

优点:

  • 高并发

  • 不会出现任何并发问题


六、PostgreSQL与MySQL隔离级别实现差异

虽然 PostgreSQL 和 MySQL 都支持相同的隔离级别名称,但实现方式完全不同。

特性 PostgreSQL MySQL(InnoDB)
默认隔离级别 Read Committed Repeatable Read
MVCC实现 tuple version undo log
Repeatable Read Snapshot Isolation MVCC + Gap Lock
Serializable SSI 强制锁
Gap Lock

七、MySQL为什么需要Gap Lock

MySQL为了防止幻读,需要:

复制代码
Gap Lock
Next-Key Lock

例如:

复制代码
SELECT * FROM user WHERE id > 10 FOR UPDATE;

MySQL会锁:

复制代码
(10, +∞)

这样其他事务无法插入新记录。

缺点:

  • 锁范围大

  • 并发性能下降


八、PostgreSQL为什么没有Gap Lock

PostgreSQL使用:

复制代码
MVCC + Snapshot

事务读取的是:

复制代码
历史版本

因此不需要锁范围来阻止插入。

优势:

  • 并发能力强

  • 不会产生大量锁


九、PostgreSQL为什么没有Read Uncommitted

SQL标准允许:

复制代码
READ UNCOMMITTED

但 PostgreSQL 中:

复制代码
READ UNCOMMITTED == READ COMMITTED

原因是:

PostgreSQL 的 MVCC 架构 天然不允许读取未提交数据

所以 PG 永远不会出现脏读


十、查看和设置隔离级别

查看当前隔离级别

复制代码
SHOW transaction_isolation;

输出示例:

复制代码
read committed

设置事务隔离级别

复制代码
SET TRANSACTION ISOLATION LEVEL REPEATABLE READ;

或者:

复制代码
BEGIN;
SET TRANSACTION ISOLATION LEVEL SERIALIZABLE;

十一、总结

PostgreSQL 的事务隔离级别总结如下:

隔离级别 脏读 不可重复读 幻读
Read Uncommitted
Read Committed
Repeatable Read
Serializable

PostgreSQL 的优势:

  • 强大的 MVCC机制

  • Repeatable Read 无幻读

  • Serializable 使用SSI并发更高

因此 PostgreSQL 在 高并发系统、金融系统、数据平台中非常受欢迎。

相关推荐
睡不醒男孩0308233 小时前
第二篇:深入探索开源数据库高可用:构建基于CLup的PostgreSQL生产级高可用与读写分离架构
数据库·postgresql·开源·clup
Micro麦可乐5 小时前
Spring Boot 实战:从零设计一个短链系统(含完整代码与数据库设计)
数据库·spring boot·后端·哈希算法·雪花算法·短链系统
码农阿豪5 小时前
从零到一:Spring Boot快速接入金仓数据库实战
数据库·spring boot·后端
鼎讯信通5 小时前
风电光缆运维提质增效:G-4000A 光缆故障追踪仪破解风场巡检难题
运维·网络·数据库
三十..6 小时前
MySQL 从入门到高可用架构实战精要
运维·数据库·mysql
cfm_29147 小时前
Redis五大基本数据结构底层了解
数据结构·数据库·redis
真实的菜7 小时前
Redis 从入门到精通(十二):典型业务场景实战 —— 排行榜、限流器、秒杀系统、Session 共享
数据库·redis·python
你想考研啊7 小时前
mysql数据库导出导入
数据库·mysql·oracle
十年编程老舅8 小时前
Linux DRM:底层逻辑与实践架构
数据库·mysql
The Sheep 20239 小时前
Vue复习
linux·服务器·数据库