CREATE TRIGGER

    注意事项

    • 当前仅支持在普通行存表上创建触发器,不支持在列存表、临时表、unlogged表等类型表上创建触发器。
    • 如果为同一事件定义了多个相同类型的触发器,则按触发器的名称字母顺序触发它们。
    • 触发器常用于多表间数据关联同步场景,对SQL执行性能影响较大,不建议在大数据量同步及对性能要求高的场景中使用。
    • 执行创建触发器操作的用户需要拥有指定表的TRIGGER权限或被授予了CREATE ANY TRIGGER权限。
    • o风格创建触发器的语法
    • 兼容mysql兼容风格的创建触发器的语法
    1. ON table_name
    2. [ FROM referenced_table_name ]
    3. { NOT DEFERRABLE | [ DEFERRABLE ] { INITIALLY IMMEDIATE | INITIALLY DEFERRED } }
    4. [ FOR [ EACH ] { ROW | STATEMENT } ]
    5. [ WHEN ( condition ) ]
    6. [ trigger_order ]
    7. trigger_body

    其中event包含以下几种:

    其中trigger_order是:

    1. { FOLLOWS|PRECEDES } other_trigger_name

    参数说明

    • CONSTRAINT

      可选项,指定此参数将创建约束触发器,即触发器作为约束来使用。除了可以使用SET CONSTRAINTS调整触发器触发的时间之外,这与常规触发器相同。 约束触发器必须是AFTER ROW触发器。

    • DEFINER

      可选项,指定此参数可以影响触发器内引用对象的权限控制。

    • IF NOT EXISTS

      可选项,指定此参数如果触发器在相同的模式中具有相同的名称、相同的表、相同的表,则防止发生错误。

    • trigger_name

      触发器名称,该名称在mysql兼容风格语法中可以指定模式,触发器模式必须与其所在表的模式相同,若不限定模式,触发器自动继承其所在表的模式,同一个表的触发器不能重名。对于约束触发器,使用SET CONSTRAINTS修改触发器行为时也使用此名称。

      取值范围:符合标识符命名规范的字符串,且最大长度不超过63个字符。

    • BEFORE

      触发器函数是在触发事件发生前执行。

    • AFTER

      触发器函数是在触发事件发生后执行,约束触发器只能指定为AFTER。

    • INSTEAD OF

      触发器函数直接替代触发事件。

    • event

      启动触发器的事件,取值范围包括:INSERT、UPDATE、DELETE或TRUNCATE,也可以通过OR同时指定多个触发事件。

      对于UPDATE事件类型,可以使用下面语法指定列:

      表示当这些列作为UPDATE语句的目标列时,才会启动触发器,但是INSTEAD OF UPDATE类型不支持指定列信息。如果UPDATE OF指定的列包含生成列,当生成列依赖的列是UPDATE语句的目标列时,也会启动触发器。

    • table_name

      需要创建触发器的表名称。

      取值范围:数据库中已经存在的表名称。

    • referenced_table_name

      约束引用的另一个表的名称。 只能为约束触发器指定,常见于外键约束。

      取值范围:数据库中已经存在的表名称。

    • DEFERRABLE | NOT DEFERRABLE

      约束触发器的启动时机,仅作用于约束触发器。这两个关键字设置该约束是否可推迟。

      详细介绍请参见。

    • INITIALLY IMMEDIATE | INITIALLY DEFERRED

      如果约束是可推迟的,则这个子句声明检查约束的缺省时间,仅作用于约束触发器。

      详细介绍请参见CREATE TABLE

    • FOR EACH ROW | FOR EACH STATEMENT

      • FOR EACH ROW是指该触发器是受触发事件影响的每一行触发一次。
      • FOR EACH STATEMENT是指该触发器是每个SQL语句只触发一次。

      未指定时默认值为FOR EACH STATEMENT。约束触发器只能指定为FOR EACH ROW。

    • condition

      决定是否实际执行触发器函数的条件表达式。当指定WHEN时,只有在条件返回true时才会调用该函数。

      在FOR EACH ROW触发器中,WHEN条件可以通过分别写入OLD.column_name或NEW.column_name来引用旧行或新行值的列。 当然,INSERT触发器不能引用OLD和DELETE触发器不能引用NEW。

      INSTEAD OF触发器不支持WHEN条件。

      WHEN表达式不能包含子查询。

      对于约束触发器,WHEN条件的评估不会延迟,而是在执行更新操作后立即发生。 如果条件返回值不为true,则触发器不会排队等待延迟执行。

    • function_name

      用户定义的函数,必须声明为不带参数并返回类型为触发器,在触发器触发时执行。

    • arguments

      执行触发器时要提供给函数的可选的以逗号分隔的参数列表。参数是文字字符串常量,简单的名称和数字常量也可以写在这里,但它们都将被转换为字符串。 请检查触发器函数的实现语言的描述,以了解如何在函数内访问这些参数。

    • trigger_order

      可选项,trigger_order特征中的{FOLLOWS|PRECEDES}控制触发器的优先触发顺序,mysql兼容性模式下允许对同一个表,在同一触发事件定义多个触发器,会按照触发器创建的先后顺序来决定触发的优先顺序(先创建的优先)。可以通过{FOLLOWS|PRECEDES}来调整优先级。使用FOLLOWS,最后一次使用的触发器与原始触发器最紧挨着,其他的触发器的优先级都顺序向后挤压;使用PRECEDES,最后一次使用的触发器与原始触发器最紧挨着,其他的触发器的优先级都顺序向前挤压。

    • trigger_body

      直接通过在begin…end之间书写代码块,定义触发器之后要完成的工作。

      表 1 表和视图上支持的触发器种类:

      表 2 PLPGSQL类型触发器函数特殊变量:

      变量名

      变量含义

      NEW

      INSERT及UPDATE操作涉及tuple信息中的新值,对DELETE为空。

      OLD

      UPDATE及DELETE操作涉及tuple信息中的旧值,对INSERT为空。

      TG_NAME

      触发器名称。

      TG_WHEN

      触发器触发时机(BEFORE/AFTER/INSTEAD OF)。

      TG_LEVEL

      触发频率(ROW/STATEMENT)。

      TG_OP

      触发操作(INSERT/UPDATE/DELETE/TRUNCATE)。

      TG_RELID

      触发器所在表OID。

      TG_RELNAME

      触发器所在表名(已废弃,现用TG_TABLE_NAME替代)。

      TG_TABLE_NAME

      触发器所在表名。

      TG_TABLE_SCHEMA

      触发器所在表的SCHEMA信息。

      TG_NARGS

      触发器函数参数个数。

      TG_ARGV[]

    1. --创建源表及触发表
    2. openGauss=# CREATE TABLE test_trigger_src_tbl(id1 INT, id2 INT, id3 INT);
    3. openGauss=# CREATE TABLE test_trigger_des_tbl(id1 INT, id2 INT, id3 INT);
    4. --创建触发器函数
    5. openGauss=# CREATE OR REPLACE FUNCTION tri_insert_func() RETURNS TRIGGER AS
    6. $$
    7. DECLARE
    8. BEGIN
    9. INSERT INTO test_trigger_des_tbl VALUES(NEW.id1, NEW.id2, NEW.id3);
    10. RETURN NEW;
    11. END
    12. $$ LANGUAGE PLPGSQL;
    13. openGauss=# CREATE OR REPLACE FUNCTION tri_update_func() RETURNS TRIGGER AS
    14. $$
    15. DECLARE
    16. BEGIN
    17. UPDATE test_trigger_des_tbl SET id3 = NEW.id3 WHERE id1=OLD.id1;
    18. RETURN OLD;
    19. END
    20. $$ LANGUAGE PLPGSQL;
    21. openGauss=# CREATE OR REPLACE FUNCTION TRI_DELETE_FUNC() RETURNS TRIGGER AS
    22. $$
    23. DECLARE
    24. BEGIN
    25. DELETE FROM test_trigger_des_tbl WHERE id1=OLD.id1;
    26. RETURN OLD;
    27. END
    28. $$ LANGUAGE PLPGSQL;
    29. openGauss=# CREATE TRIGGER insert_trigger
    30. BEFORE INSERT ON test_trigger_src_tbl
    31. FOR EACH ROW
    32. EXECUTE PROCEDURE tri_insert_func();
    33. --创建UPDATE触发器
    34. openGauss=# CREATE TRIGGER update_trigger
    35. AFTER UPDATE ON test_trigger_src_tbl
    36. FOR EACH ROW
    37. EXECUTE PROCEDURE tri_update_func();
    38. --创建DELETE触发器
    39. openGauss=# CREATE TRIGGER delete_trigger
    40. BEFORE DELETE ON test_trigger_src_tbl
    41. FOR EACH ROW
    42. EXECUTE PROCEDURE tri_delete_func();
    43. --执行INSERT触发事件并检查触发结果
    44. openGauss=# INSERT INTO test_trigger_src_tbl VALUES(100,200,300);
    45. openGauss=# SELECT * FROM test_trigger_src_tbl;
    46. openGauss=# SELECT * FROM test_trigger_des_tbl; //查看触发操作是否生效。
    47. --执行UPDATE触发事件并检查触发结果
    48. openGauss=# SELECT * FROM test_trigger_src_tbl;
    49. openGauss=# SELECT * FROM test_trigger_des_tbl; //查看触发操作是否生效
    50. --执行DELETE触发事件并检查触发结果
    51. openGauss=# DELETE FROM test_trigger_src_tbl WHERE id1=100;
    52. openGauss=# SELECT * FROM test_trigger_src_tbl;
    53. openGauss=# SELECT * FROM test_trigger_des_tbl; //查看触发操作是否生效
    54. --修改触发器
    55. openGauss=# ALTER TRIGGER delete_trigger ON test_trigger_src_tbl RENAME TO delete_trigger_renamed;
    56. --禁用insert_trigger触发器
    57. openGauss=# ALTER TABLE test_trigger_src_tbl DISABLE TRIGGER insert_trigger;
    58. --禁用当前表上所有触发器
    59. openGauss=# ALTER TABLE test_trigger_src_tbl DISABLE TRIGGER ALL;
    60. --删除触发器
    61. openGauss=# DROP TRIGGER insert_trigger ON test_trigger_src_tbl;
    62. openGauss=# DROP TRIGGER update_trigger ON test_trigger_src_tbl;
    63. openGauss=# DROP TRIGGER delete_trigger_renamed ON test_trigger_src_tbl;
    64. --创建mysql兼容数据库
    65. openGauss=# create database db_mysql dbcompatibility 'B';
    66. --创建触发器定义用户
    67. openGauss=# create user test_user password 'Gauss@123';
    68. --创建原表及触发表
    69. db_mysql=# create table test_mysql_trigger_src_tbl (id INT);
    70. db_mysql=# create table test_mysql_trigger_des_tbl (id INT);
    71. db_mysql=# create table animals (id INT, name CHAR(30));
    72. db_mysql=# create table food (id INT, foodtype VARCHAR(32), remark VARCHAR(32), time_flag TIMESTAMP);
    73. --创建MySQL兼容definer语法触发器
    74. db_mysql=# create definer=test_user trigger trigger1
    75. after insert on test_mysql_trigger_src_tbl
    76. for each row
    77. begin
    78. insert into test_mysql_trigger_des_tbl values(1);
    79. end;
    80. /
    81. --创建MySQL兼容trigger_order语法触发器
    82. db_mysql=# create trigger animal_trigger1
    83. after insert on animals
    84. for each row
    85. begin
    86. insert into food(id, foodtype, remark, time_flag) values (1,'ice cream', 'sdsdsdsd', now());
    87. end;
    88. /
    89. --创建MySQL兼容FOLLOWS触发器
    90. db_mysql=# create trigger animal_trigger2
    91. after insert on animals
    92. for each row
    93. follows animal_trigger1
    94. begin
    95. end;
    96. /
    97. db_mysql=# create trigger animal_trigger3
    98. after insert on animals
    99. for each row
    100. follows animal_trigger1
    101. begin
    102. insert into food(id, foodtype, remark, time_flag) values (3,'cake', 'sdsdsdsd', now());
    103. end;
    104. /
    105. db_mysql=# create trigger animal_trigger4
    106. for each row
    107. follows animal_trigger1
    108. begin
    109. insert into food(id, foodtype, remark, time_flag) values (4,'sausage', 'sdsdsdsd', now());
    110. end;
    111. /
    112. --执行insert触发事件并检查触发结果
    113. db_mysql=# insert into animals (id, name) values(1,'lion');
    114. db_mysql=# select * from animals;
    115. db_mysql=# select id, foodtype, remark from food;
    116. --创建MySQL兼容PROCEDES触发器
    117. db_mysql=# create trigger animal_trigger5
    118. after insert on animals
    119. for each row
    120. precedes animal_trigger3
    121. begin
    122. insert into food(id, foodtype, remark, time_flag) values (5,'milk', 'sdsds', now());
    123. end;
    124. /
    125. db_mysql=# create trigger animal_trigger6
    126. after insert on animals
    127. for each row
    128. precedes animal_trigger2
    129. begin
    130. insert into food(id, foodtype, remark, time_flag) values (6,'strawberry', 'sdsds', now());
    131. end;
    132. /
    133. --执行insert触发事件并检查触发结果
    134. db_mysql=# insert into animals (id, name) values(2, 'dog');
    135. db_mysql=# select * from animals;
    136. db_mysql=# select id, foodtype, remark from food;
    137. --创建MySQL兼容if not exists语法触发器
    138. db_mysql=# create trigger if not exists animal_trigger1
    139. after insert on animals
    140. for each row
    141. begin
    142. insert into food(id, foodtype, remark, time_flag) values (1,'ice cream', 'sdsdsdsd', now());
    143. end;
    144. /
    145. --mysql兼容删除触发器语法
    146. db_mysql=# drop trigger animal_trigger1;
    147. db_mysql=# drop trigger if exists animal_trigger1;
    148. --在指定模式下创建、重命名、删除触发器语法,触发器的模式需要与表模式相同
    149. create schema testscm;
    150. create table food (id int, foodtype varchar(32), remark varchar(32), time_flag timestamp);
    151. create table testscm.animals_scm (id int, name char(30));
    152. -- 在指定模式下创建触发器
    153. create trigger testscm.animals_trigger
    154. after insert on testscm.animals_scm
    155. for each row
    156. begin
    157. insert into food(id, foodtype, remark, time_flag) values (1,'bamboo', 'healthy', now());
    158. end;
    159. /
    160. create trigger if not exists testscm.animals_trigger
    161. after insert on testscm.animals_scm
    162. for each row
    163. begin
    164. insert into food(id, foodtype, remark, time_flag) values (1,'bamboo', 'healthy', now());
    165. end;
    166. /
    167. -- 重命名指定模式下的触发器
    168. alter trigger testscm.animals_trigger on testscm.animals_scm rename to animals_trigger_new;
    169. -- 删除指定模式下的触发器
    170. drop trigger if exists testscm.animals_trigger_new;

    相关链接

    DROP TRIGGER