PostgreSQL 约束延迟触发

PG只支持以下几种约束延迟触发(并不是不触发,而是再事务结束的时候触发)

UNIQUEPRIMARY KEYREFERENCES(外键)和 EXCLUDE

测试例子:

sql 复制代码
CREATE TABLE logs (
    id INTEGER,
    log_time TIMESTAMP,
    message TEXT,
    CONSTRAINT pk_logs 
        PRIMARY KEY (id) 
        DEFERRABLE
);

BEGIN;
SET CONSTRAINTS pk_logs DEFERRED;

-- 可以暂时插入重复ID,最后再修复
INSERT INTO logs (id, log_time, message) VALUES
(1, '2024-01-01 10:00:00', 'Message 1'),
(1, '2024-01-01 10:01:00', 'Message 2'), -- 暂时允许重复
(2, '2024-01-01 10:02:00', 'Message 3');

-- 修复重复的ID
UPDATE logs SET id = 3 WHERE id = 1 AND message = 'Message 2';

COMMIT;  -- 提交时才检查主键唯一性


建表的时候指定DEFERRABLE是可以延迟的意思,是延迟的前提条件,默认是 INITIALLY IMMEDIATE

这时候需要在开始事务之后使用
SET CONSTRAINTS DEFERRED语句来确定是否要真的延迟

当然也可以指定INITIALLY DEFERRED 这样在事务中就不需要执行
SET CONSTRAINTS DEFERRED 语句了

如下图所测试

指定INITIALLY DEFERRED

sql 复制代码
CREATE TABLE logs (
    id INTEGER,
    log_time TIMESTAMP,
    message TEXT,
    CONSTRAINT pk_logs 
        PRIMARY KEY (id) 
        DEFERRABLE INITIALLY DEFERRED
);

begin;

INSERT INTO logs (id, log_time, message) VALUES
(1, '2024-01-01 10:00:00', 'Message 1'),
(1, '2024-01-01 10:01:00', 'Message 2'), 
(2, '2024-01-01 10:02:00', 'Message 3');

-- 修复重复的ID
UPDATE logs SET id = 3 WHERE id = 1 AND message = 'Message 2';

COMMIT;  -- 提交时才检查主键唯一性

修改约束

当然也可以修改约束延迟模式

但是需要先删除掉约束,重建的时候指定

sql 复制代码
postgres=# alter table logs drop constraint pk_logs;
ALTER TABLE
postgres=# alter table logs add constraint pk_logs primary key(id) deferrable initially deferred;
ALTER TABLE

当然oracle和达梦默认就是延迟模式

如下所示:

sql 复制代码
Connected to:
Oracle Database 19c Enterprise Edition Release 19.0.0.0.0 - Production
Version 19.18.0.0.0

SQL> create table tab(id numeric primary key);

Table created.

SQL> insert into tab values (1);

1 row created.

SQL> insert into tab values(2);

1 row created.

SQL> commit;

Commit complete.

SQL> update tab set id=id+1;

2 rows updated.

SQL> commit;

Commit complete.

SQL>

达梦

sql 复制代码
SQL> create table tab(id int);
executed successfully
used time: 4.234(ms). Execute id is 1702.
SQL> alter table tab add constraint tab_pkey primary key(id);
executed successfully
used time: 12.044(ms). Execute id is 1703.
SQL> insert into tab values (1),(2);
affect rows 2

used time: 1.255(ms). Execute id is 1704.
SQL> commit;
executed successfully
used time: 1.605(ms). Execute id is 1705.
SQL> select * from v$version;

LINEID     BANNER
---------- ---------------------------------
1          DM Database Server 64 V8
2          DB Version: 0x7000c
3          03134284194-20240703-234060-20108
4          Msg Version: 12
5          Gsu level(5) cnt: 0

used time: 0.336(ms). Execute id is 1706.
SQL> update tab set id=id+1;
affect rows 2

used time: 1.135(ms). Execute id is 1707.
SQL> commit;
executed successfully
used time: 1.596(ms). Execute id is 1708.
SQL>
相关推荐
llilian_161 小时前
IRIG-B码产生器立足用户痛点,提供精准授时解决方案
大数据·数据库·功能测试·单片机·嵌入式硬件·测试工具
zuoerjinshu6 小时前
sql实战解析-sum()over(partition by xx order by xx)
数据库·sql
NocoBase7 小时前
【2.0 教程】第 1 章:认识 NocoBase ,5 分钟跑起来
数据库·人工智能·开源·github·无代码
Hoshino.419 小时前
基于Linux中的数据库操作——下载与安装(1)
linux·运维·数据库
Oueii10 小时前
Django全栈开发入门:构建一个博客系统
jvm·数据库·python
未来龙皇小蓝11 小时前
【MySQL-索引调优】11:Group by相关概念
数据库·mysql·性能优化
2401_8318249611 小时前
使用Fabric自动化你的部署流程
jvm·数据库·python
njidf11 小时前
Python日志记录(Logging)最佳实践
jvm·数据库·python
twc82911 小时前
大模型生成 QA Pairs 提升 RAG 应用测试效率的实践
服务器·数据库·人工智能·windows·rag·大模型测试
@我漫长的孤独流浪11 小时前
Python编程核心知识点速览
开发语言·数据库·python