如果列之一已更改,PostgreSQL将触发

koa*_*ala 3 postgresql triggers

如果至少一个字段发生了更改,我想跟踪表上的更改并将它们存储在相应的历史记录表中。

我有以下表格:

tbl_employees

CREATE TABLE tbl_employees (
    id serial NOT NULL,
    first_name character varying NOT NULL,
    last_name character varying NOT NULL
)
Run Code Online (Sandbox Code Playgroud)

tbl_employees_history

CREATE TABLE tbl_employees (
    id serial NOT NULL,
    employee_id NOT NULL,
    first_name character varying NOT NULL,
    last_name character varying NOT NULL,
    changed_on timestamp NOT NULL
)
Run Code Online (Sandbox Code Playgroud)

功能

CREATE OR REPLACE FUNCTION log_employee_changes() RETURNS trigger as 
$BODY$
BEGIN
     IF (NEW.last_name <> OLD.last_name) or (NEW.first_name <> OLD.first_name) THEN
     INSERT INTO tbl_employee_history(employee_id,last_name, first_name,changed_on)
     VALUES(OLD.id,OLD.last_name, OLD.first_name,now());
     END IF;
RETURN NEW;
END;
$BODY$
LANGUAGE plpgsql;
Run Code Online (Sandbox Code Playgroud)

触发

CREATE TRIGGER add_log
BEFORE UPDATE ON tbl_employees
FOR EACH ROW
EXECUTE PROCEDURE log_employee_changes();
Run Code Online (Sandbox Code Playgroud)

到目前为止,一切都很好。但是我想知道我是否有一个可以说15列的表,并且如果其中一列已更改,我想保存历史记录。我是否必须为每个字段一遍又一遍地重复IF语句,还是有某种技巧可以只检查每一列?这样,我就可以在表中添加一列,而不必忘记更改触发器功能。

像这样:

CREATE OR REPLACE FUNCTION log_employee_changes() RETURNS trigger as 
$BODY$
BEGIN
     IF ?????there is one change on whatever column????? THEN
     INSERT INTO tbl_employee_history(employee_id,last_name, first_name,changed_on)
     VALUES(OLD.id,OLD.last_name, OLD.first_name,now());
     END IF;
RETURN NEW;
END;
$BODY$
LANGUAGE plpgsql;
Run Code Online (Sandbox Code Playgroud)

kli*_*lin 5

您可以比较整个记录:

if new <> old then ...
Run Code Online (Sandbox Code Playgroud)

要么

if new is distinct from old then ...
Run Code Online (Sandbox Code Playgroud)

第二种选择更为笼统。仅当确定记录不能包含空值时才使用第一个。