深入理解PostgreSQL 事务处理

 更新时间:2025年06月16日 09:37:26   作者:熊大如如  
PostgreSQL事务处理确保数据一致性,支持四种隔离级别,处理脏读、不可重复读和幻读问题,并可通过会话或配置文件设置,自动提交默认开启,感兴趣的可以了解一下

简介

PostgreSQL 事务处理(Transaction Processing)是指在数据库中执行一系列 SQL 语句,使其成为一个不可分割的操作单元,即 要么全部执行成功,要么全部回滚,以确保数据的一致性和完整性

准备工作

  • 创建演示表
CREATE TABLE "public"."users" (
  "user_account" varchar(32) COLLATE "pg_catalog"."default" NOT NULL,
  "username" varchar(32) COLLATE "pg_catalog"."default",
  "user_avatar" varchar(64) COLLATE "pg_catalog"."default",
  "user_profile" varchar(512) COLLATE "pg_catalog"."default",
  "hashed_password" varchar COLLATE "pg_catalog"."default" NOT NULL
);

基本操作

  • 提交事务
BEGIN; -- 开启事务

-- SQL 语句
INSERT INTO users (user_account, hashed_password) VALUES ('Alice', 'xxxx');

COMMIT; -- 提交事务
  • 回滚事务
BEGIN

DELETE FROM users;

ROLLBACK;
  • 设置回滚点

SAVEPOINT 允许在事务内部创建回滚点,部分 SQL 语句可以回滚,而不影响其他 SQL

BEGIN;

INSERT INTO users (user_account, hashed_password) VALUES ('Alice', 'xxxx');
SAVEPOINT sp1; -- 创建回滚点

INSERT INTO users (user_account, hashed_password) VALUES ('Alice2', 'xxxx');
SAVEPOINT sp2;

INSERT INTO users (user_account, hashed_password) VALUES ('Alice3', 'xxxx');
ROLLBACK TO sp2;

COMMIT; -- 提交事务

事务隔离级别

简介

在数据库中,事务隔离级别(Transaction Isolation Levels)用于控制多个事务并发执行时的可见性,避免数据不一致的问题。PostgreSQL 遵循 ACID(原子性、一致性、隔离性、持久性) 原则,并提供四种事务隔离级别

隔离级别脏读不可重复读幻读
读未提交✅ 可能发生✅ 可能发生✅ 可能发生
读已提交(默认)❌ 不会发生✅ 可能发生✅ 可能发生
可重复读❌ 不会发生❌ 不会发生✅ 可能发生
可串行化❌ 不会发生❌ 不会发生❌ 不会发生

下面我们来逐一介绍

读未提交/读已提交

PostgreSQL 不真正支持 读未提交 这个级别,而是当作 **读已提交 **处理

即,就算你设置了这个级别,PG 数据库还是会使用 读已提交 级别事务隔离

脏读示例(PG 不支持)

BEGIN; -- 事务1
SET TRANSACTION ISOLATION LEVEL READ UNCOMMITTED;
UPDATE users SET hashed_password = 'new_hash' WHERE user_account = 'Alice';
SELECT txid_current(); -- 查看当前事务id
-- 保持事务未提交

BEGIN; -- 事务2
SET TRANSACTION ISOLATION LEVEL READ UNCOMMITTED;
SELECT * FROM users WHERE user_account = 'Alice';
SELECT txid_current();


ROLLBACK; -- 释放锁,事务 2 的查询继续执行

注意:我的 navacat17 版本,一个查询页面执行的语句是同一个事务,所以想让上面语句生效,你可能需要开启两个查询页面,分别执行事务 1 和 2

不可重复读

  • 事务 1
BEGIN;
SELECT hashed_password FROM users;
  • 事务 2
BEGIN;
UPDATE users SET hashed_password = 'xxxx';
COMMIT;
  • 事务 1
SELECT hashed_password FROM users;

问题:事务 1 在第一次 SELECT 时看到的是 hashed_password 与 第二次查询时hashed_password 不一致,这就是不可重复读

幻读

  • 事务 1
BEGIN;
SELECT COUNT(*) FROM users; -- 假设是3
  • 事务 2
BEGIN;
INSERT INTO users (user_account, hashed_password) VALUES ('Alice4', 'xxxx');
COMMIT; -- 增加到4
  • 事务 1
SELECT COUNT(*) FROM users; -- 增加到4

问题:事务 1 在开始时认为 users 里数据为 3,但在事务进行中,别的事务插入了一条数据,事务 1 重新查询时,发现数据数量变了,这就是幻读

其余隔离级别,有需求可自行查询~

设置事务隔离级别

在事务中设置

BEGIN;
SET TRANSACTION ISOLATION LEVEL REPEATABLE READ;
-- SQL 语句
COMMIT;

在会话级别设置

SET SESSION CHARACTERISTICS AS TRANSACTION ISOLATION LEVEL SERIALIZABLE;
  • 这会影响当前会话中的所有事务

在 PostgreSQL 配置文件 postgresql.conf 设置(全局)

default_transaction_isolation = 'read committed'
  • 影响所有数据库的默认隔离级别

自动提交

PostgreSQL 默认开启自动提交模式,即每条 SQL 语句都会被自动提交。如果要手动管理事务,需要显式使用 BEGIN

SET AUTOCOMMIT TO OFF;

到此这篇关于深入理解PostgreSQL 事务处理的文章就介绍到这了,更多相关PostgreSQL 事务处理内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家! 

相关文章

  • PostgreSQL游标与索引选择实例详细介绍

    PostgreSQL游标与索引选择实例详细介绍

    这篇文章主要介绍了PostgreSQL游标与索引选择优化案例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习吧
    2022-09-09
  • PostgreSQL数据DML误操作恢复方法

    PostgreSQL数据DML误操作恢复方法

    PostgreSQL是一种开源的对象关系型数据库管理系统,其DML(Data Manipulation Language)负责数据的操作和管理,那么如何DML误操作了如何恢复,所以本文介绍了PostgreSQL数据DML误操作恢复方法,需要的朋友可以参考下
    2024-12-12
  • 使用docker compose启动postgresql的示例代码

    使用docker compose启动postgresql的示例代码

    要在启动 PostgreSQL 容器时执行特定的初始化文件,可以使用 Docker 的 docker-entrypoint-initdb.d 目录,这个目录下的 SQL 文件会在容器启动时被自动执行,下面是如何修改 Docker Compose 配置文件,以便在启动时执行初始化 SQL 脚本,需要的朋友可以参考下
    2024-10-10
  • 解决postgresql 数字转换成字符串前面会多出一个空格的问题

    解决postgresql 数字转换成字符串前面会多出一个空格的问题

    这篇文章主要介绍了解决postgresql 数字转换成字符串前面会多出一个空格的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-12-12
  • PostgreSQL之连接失败的问题及解决

    PostgreSQL之连接失败的问题及解决

    这篇文章主要介绍了PostgreSQL之连接失败的问题及解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-05-05
  • PostgreSQL设置主键自增的方法详解

    PostgreSQL设置主键自增的方法详解

    在使用Mysql时,创建表结构时可以通过关键字auto_increment来指定主键是否自增,但在Postgresql数据库中,虽然可以实现字段的自增,但从本质上来说却并不支持Mysql那样的自增,所以本文给大家介绍了PostgreSQL如何设置主键自增,需要的朋友可以参考下
    2024-08-08
  • 使用PostgreSQL创建高级搜索引擎的代码示例

    使用PostgreSQL创建高级搜索引擎的代码示例

    本文我们将探索PostgreSQL中的全文搜索功能,并研究我们能够复制多少典型搜索引擎功能,文中有详细的代码示例供大家参考,需要的朋友可以参考下
    2023-07-07
  • PostgreSQL无法查看表中数据问题的原因和解决方法

    PostgreSQL无法查看表中数据问题的原因和解决方法

    在 PostgreSQL 中,恢复数据库后,使用 DBeaver 等工具可以看到数据库和表名,但无法查询到表中数据,可能有很多原因,本文给大家分析了PostgreSQL无法查看表中数据问题的原因和解决方法,需要的朋友可以参考下
    2025-04-04
  • PostgreSQL创建新用户所遇见的权限问题以及解决办法

    PostgreSQL创建新用户所遇见的权限问题以及解决办法

    这篇文章主要给大家介绍了关于PostgreSQL创建新用户所遇见的权限问题以及解决办法, 在PostgreSQL中创建一个新用户非常简单,但可能会遇到权限问题,需要的朋友可以参考下
    2023-09-09
  • PostgreSQL对GROUP BY子句使用常量的特殊限制详解

    PostgreSQL对GROUP BY子句使用常量的特殊限制详解

    这篇文章主要介绍了PostgreSQL对GROUP BY子句使用常量的特殊限制详解,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-02-02

最新评论