[英]PostgreSQL Update TRIGGER Fires Multiple Times When Just 1 Row Updated
源表:-
CREATE TABLE schema1.Source_Table
(
Source_Table_id serial NOT NULL,
current_status_id smallint NOT NULL,
current_status_reason varchar(200) NULL,
requestor_id integer NOT NULL,
approver_id integer NULL,
last_upd_user_id integer NOT NULL,
last_upd_date_time timestamp without time zone DEFAULT CURRENT_TIMESTAMP NOT NULL,
CONSTRAINT PK_Source_Table PRIMARY KEY (Source_Table_id)
)
WITH OIDS;
目标表(审计历史目的):-
CREATE TABLE schema2.Destination_Table
(
type_id smallint NOT NULL,
id integer NOT NULL,
state_id smallint NOT NULL,
state_reason varchar(200) NULL,
requestor_id integer NOT NULL,
approver_id integer NULL,
upd_by_user_id integer NOT NULL,
upd_by_user_type smallint NOT NULL,
upd_date_time timestamp without time zone NOT NULL
)
WITH OIDS;
更新源表上的每个行触发器后:-
CREATE TRIGGER trg_upd_Source_Table
AFTER UPDATE of current_status_id
ON schema1.Source_Table
FOR EACH ROW
WHEN (OLD.current_status_id IS DISTINCT FROM NEW.current_status_id)
EXECUTE PROCEDURE schema1.Source_Table_hist();
上面每个行触发器的更新后的触发器函数:-
CREATE OR REPLACE FUNCTION schema1.Source_Table_hist()
RETURNS TRIGGER
LANGUAGE PLPGSQL
AS $$
BEGIN
INSERT INTO schema2.Destination_Table
(type_id, id, state_id, state_reason, requestor_id, approver_id, upd_by_user_id,
upd_by_user_type, upd_date_time)
SELECT 1, OLD.Source_Table_id, OLD.current_status_id, OLD.current_status_reason,
OLD.requestor_id, OLD.approver_id, OLD.last_upd_user_id, 1, OLD.last_upd_date_time
from schema1.Source_Table
where OLD.current_status_id IS DISTINCT FROM NEW.current_status_id;
RETURN NULL;
END;
$$
schema1.Source_Table 表中已经有 8 行具有唯一的主键 Source_Table_id。 当我使用主键更新此表的 1 行时,如下所示,它将 8 行(1 个原始行和 7 个副本)插入到 schema2.Destination_Table 表中,而不是仅插入 1 行。
update schema1.Source_Table
set current_status_id = 4
where Source_Table_id = 9;
这里的问题是:-当仅更新该表的 1 行时,为什么触发器会触发 8 次(这等于在其上创建此触发器的表中的总行数)。
预期行为:-触发器应仅触发一次,然后在创建触发器的源表中仅更新 1 行时,在目标审计表中插入 1 行。
如何解决这个问题?
触发器不会多次触发,您的查询正在为源表中的每一行在 hist 表中插入一行:
INSERT INTO schema2.Destination_Table
(type_id, id, state_id, state_reason, requestor_id, approver_id, upd_by_user_id,
upd_by_user_type, upd_date_time)
SELECT 1, OLD.Source_Table_id, OLD.current_status_id, OLD.current_status_reason,
OLD.requestor_id, OLD.approver_id, OLD.last_upd_user_id, 1, OLD.last_upd_date_time
from schema1.Source_Table
where OLD.current_status_id IS DISTINCT FROM NEW.current_status_id;
RETURN NULL;
我认为你不需要那个 from 子句。
问题出在 WHERE 条件中:
where OLD.current_status_id IS DISTINCT FROM NEW.current_status_id
根据触发器上的 WHEN 条件,已知此条件为真。 因为它是唯一有效的 WHERE,所以完全没有 WHERE 条件,因此所有鱼子都由插入物处理。 建议
where current_status_id = OLD.current_status_id
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.