如果在事务内部调用了触发器,则可以保证在事务中的下一个查询开始之前完成所有触发器(非约束/非延迟触发器)。 例如,我有以下表格:
Table a
id name is_updated
1 alice False
2 alice False
3 Alice False
在此表上定义触发器之前的行级别,如果名称字段已更新,则将is_updated设置为True。
我执行以下交易:
BEGIN;
UPDATE table_a SET name = 'Alice' WHERE name = 'alice';
UPDATE table_a SET is_updated = False WHERE is_updated = True;
COMMIT;
是否可以在所有触发器完成之前启动第二个UPDATE语句?或者触发器在带有调用触发器的语句的内部运行?
UPD1
Postgresql版本是9.3。 这里是触发器和相关的功能源:
CREATE TRIGGER a_name_update_trigger
BEFORE UPDATE OF name
FOR EACH ROW
EXECUTE PROCEDURE on_a_name_update();
CREATE OR REPLACE FUNCTION on_a_name_update()
RETURNS TRIGGER AS $$
BEGIN
NEW.is_updated = True;
RETURN NEW;
END;
$$ LANGUAGE plpgsql;
答案 0 :(得分:3)
没有什么是错的,除了我认为触发器的定义应该是这样的:
CREATE OR REPLACE FUNCTION on_a_name_update()
RETURNS TRIGGER AS $$
BEGIN
NEW.is_updated = true;
RETURN NEW;
END;
$$ LANGUAGE plpgsql;
CREATE TRIGGER a_name_update_trigger
BEFORE update of name ON table_a
FOR EACH ROW
EXECUTE PROCEDURE on_a_name_update();
所以,通过执行你的查询:
BEGIN;
UPDATE table_a SET name = 'Alice' WHERE name = 'alice';
UPDATE table_a SET is_updated = False WHERE is_updated = True;
COMMIT;
最终更新符合预期:
id name is_updated
1 "Alice" False
2 "Alice" False
3 "Alice" False
换句话说,第二次更新是在对行触发触发器后执行的(如果col名称已更新)