0

0

postgresql约束触发顺序如何决定_postgresql约束执行模型

冷漠man

冷漠man

发布时间:2025-12-02 22:59:02

|

533人浏览过

|

来源于php中文网

原创

PostgreSQL约束触发顺序由系统按阶段和类型自动决定:语句前检查NOT NULL(早于CHECK),语句中检查UNIQUE/PRIMARY KEY/FOREIGN KEY,语句结束时检查DEFERRABLE约束;同阶段内除NOT NULL优先外无确定顺序,延迟约束仅推迟检查时机,不改变语义;触发器在约束前运行,可用于预处理数据。

postgresql约束触发顺序如何决定_postgresql约束执行模型

PostgreSQL 中约束的触发顺序不是由用户显式指定的,而是由系统根据约束类型、定义方式和执行阶段自动决定的。理解这个顺序的关键,在于区分“约束检查时机”(timing)和“约束检查顺序”(ordering)——前者由 DEFERRABLEINITIALLY 属性控制,后者在同阶段内通常无严格保证(除个别例外)。

约束按执行阶段分层,而非全局排序

PostgreSQL 把约束检查嵌入到 DML 执行的固定生命周期中,大致分为三个阶段:

  • 语句开始前:仅检查 NOT NULLCHECK(如果定义为 NOT DEFERRABLE)在行插入/更新时的即时值,但实际发生在“行修改前”,属于最前置校验;
  • 语句执行过程中:唯一键(UNIQUE)、主键(PRIMARY KEY)、外键(FOREIGN KEY)的“非延迟”检查默认在此阶段进行,即每行修改后立即验证(例如插入一行就查一次索引是否冲突);
  • 语句结束时(或事务提交时):仅适用于显式声明为 DEFERRABLEINITIALLY DEFERRED 的约束,它们被推迟到 COMMIT 前统一检查。

同阶段内,多数约束无确定顺序

对于同一语句中多个 NOT DEFERRABLE 约束(如多个 CHECK 或一个 UNIQUE 加一个 FOREIGN KEY),PostgreSQL 不保证它们的检查先后。例如:

CREATE TABLE t (a INT CHECK (a > 0), b INT UNIQUE);
插入 (−1, 1) 时,可能先报 CHECK 错误,也可能先报 UNIQUE 冲突(取决于内部实现细节和索引扫描路径),不应依赖特定顺序。

但有一个明确例外:NOT NULL 总是早于 CHECK 执行。因为 NOT NULL 是存储层硬性要求,而 CHECK 是逻辑层表达式,后者依赖前者已通过的非空值。

Runway Green Screen
Runway Green Screen

Runway 平台的AI视频工具,绿幕抠除、视频生成、动态捕捉等

下载

延迟约束(DEFERRABLE)可跨语句重排检查时机

延迟约束不改变“类型优先级”,而是把检查从语句级推迟到事务级,从而允许临时违反约束(只要最终满足)。例如:

  • 两个互为外键的表,需先插入一方再插入另一方——若外键设为 DEFERRABLE INITIALLY DEFERRED,就能在单个事务中完成两步插入,最后 COMMIT 时统一校验;
  • 使用 SET CONSTRAINTS ... DEFERRED 可在运行时动态切换,让原本 INITIALLY IMMEDIATE 的约束临时延迟;
  • 注意:延迟约束只影响“检查时间点”,不改变约束本身语义;它也不能绕过 NOT NULL 或即时唯一性(如序列冲突)等底层限制。

触发器与约束的协作关系

约束本身不触发触发器,但触发器(BEFORE ROW)会在约束检查之前运行。这意味着:

  • 你可以在 BEFORE INSERT 触发器中修改新行(NEW),从而影响后续约束检查结果(例如自动补全 NOT NULL 字段);
  • 反之,若触发器抛出异常,约束根本不会执行;
  • 因此,若需强顺序控制,应优先用触发器预处理数据,再交由约束做终审——这是比依赖约束顺序更可靠的做法。

基本上就这些。PostgreSQL 的约束模型本质是“分阶段 + 类型驱动”,不是靠用户排序,而是靠合理选择 DEFERRABLE、配合触发器、并理解各约束的底层机制来达成预期行为。

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

231

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

436

2024.03.01

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

316

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

538

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

52

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

postgresql常用命令
postgresql常用命令

postgresql常用命令psql、createdb、dropdb、createuser、dropuser、l、c、dt、d table_name、du、i file_name、e和q等。本专题为大家提供postgresql相关的文章、下载、课程内容,供大家免费下载体验。

158

2023.10.10

常用的数据库软件
常用的数据库软件

常用的数据库软件有MySQL、Oracle、SQL Server、PostgreSQL、MongoDB、Redis、Cassandra、Hadoop、Spark和Amazon DynamoDB。更多关于数据库软件的内容详情请看本专题下面的文章。php中文网欢迎大家前来学习。

970

2023.11.02

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

26

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号