在PostgreSQL的实际业务开发中,经常需要实现当表中某个字段的值发生变化时,才执行特定的业务逻辑,比如用户余额更新时记录流水、订单状态变更时发送通知等。如果使用普通的触发器,只要触发事件(如UPDATE)发生就会执行逻辑,哪怕变化的字段和需要监听的字段无关,这会造成不必要的性能开销。而PostgreSQL的触发器支持WHEN子句,可以精准过滤触发条件,仅当满足指定字段变化规则时才执行触发逻辑。

WHEN子句的基本语法
PostgreSQL的触发器定义中,WHEN子句用于指定触发器的执行条件,只有满足条件时才会执行触发器函数。其基本语法结构如下:
CREATE TRIGGER trigger_name
{BEFORE | AFTER} {INSERT | UPDATE | DELETE} [OR ...]
ON table_name
[FROM referenced_table_name]
FOR EACH {ROW | STATEMENT}
[WHEN ( condition )]
EXECUTE FUNCTION function_name();
其中condition是判断条件,可以使用OLD和NEW伪记录来引用更新前后的字段值,仅当条件返回真时,才会执行后续的触发器函数。
基于字段变化的常见使用场景
- 仅当特定字段被更新时执行逻辑,忽略其他字段的更新操作
- 仅当字段值变为指定内容时触发,比如订单状态变为已支付时记录日志
- 仅当字段值变化前后满足特定对比关系时触发,比如余额减少时触发预警
具体实现示例
场景1:仅当指定字段更新时触发
假设有一张用户表users,结构如下:
CREATE TABLE users (
id INT PRIMARY KEY,
username VARCHAR(50),
balance DECIMAL(10,2),
last_login_time TIMESTAMP
);
现在需要实现一个触发器,仅当用户余额balance字段被更新时,才记录一条余额变更日志到balance_log表,其他字段更新时不触发。首先创建日志表:
CREATE TABLE balance_log (
id SERIAL PRIMARY KEY,
user_id INT,
old_balance DECIMAL(10,2),
new_balance DECIMAL(10,2),
change_time TIMESTAMP DEFAULT CURRENT_TIMESTAMP
);
然后创建触发器函数:
CREATE OR REPLACE FUNCTION log_balance_change()
RETURNS TRIGGER AS $$
BEGIN
INSERT INTO balance_log (user_id, old_balance, new_balance)
VALUES (NEW.id, OLD.balance, NEW.balance);
RETURN NEW;
END;
$$ LANGUAGE plpgsql;
最后创建带WHEN子句的触发器,仅当balance字段发生变化时触发:
CREATE TRIGGER balance_update_trigger AFTER UPDATE ON users FOR EACH ROW WHEN (OLD.balance IS DISTINCT FROM NEW.balance) EXECUTE FUNCTION log_balance_change();
这里使用IS DISTINCT FROM来判断字段值是否发生变化,避免NULL值导致的判断异常。如果执行更新其他字段的语句,比如更新用户名:
UPDATE users SET username = 'new_name' WHERE id = 1;
此时balance字段没有变化,触发器不会执行,不会插入日志。如果更新余额:
UPDATE users SET balance = balance - 100 WHERE id = 1;
此时balance字段发生变化,触发器会执行,插入对应的变更日志。
场景2:仅当字段值变为指定内容时触发
假设有一张订单表orders,结构如下:
CREATE TABLE orders (
order_id INT PRIMARY KEY,
order_status VARCHAR(20),
order_amount DECIMAL(10,2)
);
需要实现当订单状态order_status从其他状态变为已支付时,自动记录支付时间到pay_time字段(假设表中新增了该字段):
ALTER TABLE orders ADD COLUMN pay_time TIMESTAMP;
创建触发器函数:
CREATE OR REPLACE FUNCTION set_pay_time()
RETURNS TRIGGER AS $$
BEGIN
NEW.pay_time = CURRENT_TIMESTAMP;
RETURN NEW;
END;
$$ LANGUAGE plpgsql;
创建触发器,仅当order_status变为已支付时触发:
CREATE TRIGGER order_paid_trigger BEFORE UPDATE ON orders FOR EACH ROW WHEN (NEW.order_status = '已支付' AND OLD.order_status <> '已支付') EXECUTE FUNCTION set_pay_time();
当执行更新语句将订单状态改为已支付时:
UPDATE orders SET order_status = '已支付' WHERE order_id = 1;
触发器会执行,自动设置pay_time为当前时间。如果订单状态只是从已支付改为已发货,或者更新其他字段,触发器都不会执行。
使用注意事项
- WHEN子句中只能引用
OLD和NEW伪记录,不能使用子查询或者其他表的数据 OLD伪记录在INSERT触发器中不可用,NEW伪记录在DELETE触发器中不可用,WHEN子句的条件需要匹配触发事件类型- 判断字段是否变化时,建议使用
IS DISTINCT FROM而不是<>,因为<>在字段值为NULL时无法正确判断 - WHEN子句的条件判断是在触发器函数执行之前进行的,符合条件才会调用函数,因此可以减少不必要的函数执行开销
总结
PostgreSQL的触发器WHEN子句是实现基于字段变化精准触发逻辑的有效方式,通过合理设置条件,可以避免不必要的触发器执行,提升数据库性能。开发者可以根据实际业务场景,灵活组合OLD和NEW的字段值判断,实现各种复杂的触发条件过滤,满足业务需求。
PostgreSQLtriggerWHEN_clause字段变化触发修改时间:2026-06-10 03:30:20