返回首页

gbase数据、南大通用产品文档:GBase8cCREATE TRIGGER

更新日期:2024年09月11日

功能描述
创建一个触发器。触发器将与指定的表或视图关联,
并在特定条件下执行指定的函数。
注意事项
当前仅支持在普通行存表上创建触发器,不支持在列存表、临时表、unlogged 表等类型
表上创建触发器。
如果为同一事件定义了多个相同类型的触发器,则按触发器的名称字母顺序触发它们。
触发器常用于多表间数据关联同步场景,对SQL 执行性能影响较大,不建议在大数据
量同步及对性能要求高的场景中使用。
语法格式

GBase 8c SQL 参考手册
南大通用数据技术股份有限公司
1067
CREATE [ CONSTRAINT ] TRIGGER trigger_name { BEFORE | AFTER | INSTEAD OF }
{ event [ OR ... ] }
ON table_name
[ FROM referenced_table_name ]
{ NOT DEFERRABLE | [ DEFERRABLE ] { INITIALLY IMMEDIATE | INITIALLY
DEFERRED } }
[ FOR [ EACH ] { ROW | STATEMENT } ]
[ WHEN ( condition ) ]
EXECUTE PROCEDURE function_name ( arguments );
其中event 包含以下几种:
INSERT
UPDATE [ OF column_name [, ... ] ]
DELETE
TRUNCATE
参数说明

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

trigger_name
触发器名称,该名称不能限定模式,
因为触发器自动继承其所在表的模式,且同一个表
的触发器不能重名。对于约束触发器,
使用SET CONSTRAINTS 修改触发器行为时也使用
此名称。
取值范围:符合标识符命名规范的字符串,且最大长度不超过63 个字符。

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

GBase 8c SQL 参考手册
南大通用数据技术股份有限公司
1068

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

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

event
启动触发器的事件,取值范围包括:INSERT、UPDATE、DELETE 或TRUNCATE,也
可以通过OR 同时指定多个触发事件。
对于UPDATE 事件类型,可以使用下面语法指定列:
UPDATE OF column_name1 [, column_name2 ... ]
表示当这些列作为UPDATE 语句的目标列时,才会启动触发器,但是INSTEAD OF
UPDATE 类型不支持指定列信息。

table_name
需要创建触发器的表名称。
取值范围:数据库中已经存在的表名称。

referenced_table_name
约束引用的另一个表的名称。只能为约束触发器指定,常见于外键约束。
取值范围:数据库中已经存在的表名称。

DEFERRABLE | NOT DEFERRABLE
约束触发器的启动时机,仅作用于约束触发器。这两个关键字设置该约束是否可推迟。
详细介绍请参见CREATE TABLE。

INITIALLY IMMEDIATE | INITIALLY DEFERRED
如果约束是可推迟的,则这个子句声明检查约束的缺省时间,仅作用于约束触发器。
详细介绍请参见CREATE TABLE。

GBase 8c SQL 参考手册
南大通用数据技术股份有限公司
1069

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

GBase 8c SQL 参考手册
南大通用数据技术股份有限公司
1070
INSTEAD OF 的触发器必须标记为FOR EACH ROW,并且只能在视图上定义。
BEFORE 和AFTER 触发器作用在视图上时,只能标记为FOR EACH STATEMENT。
TRUNCATE 类型触发器仅限FOR EACH STATEMENT。
表1 表和视图上支持的触发器种类:
触发时机
触发事件
行级
语句级
BEFORE
INSERT/UPDATE/DELET
E

表和视图
TRUNCATE
不支持

AFTER
INSERT/UPDATE/DELET
E

表和视图
TRUNCATE
不支持

INSTEAD O
F
INSERT/UPDATE/DELET
E
视图
不支持
TRUNCATE
不支持
不支持
表2 PLPGSQL 类型触发器函数特殊变量:
变量名
变量含义
NEW
INSERT 及UPDATE 操作涉及tuple 信息中
的新值,对DELETE 为空。
OLD
UPDATE 及DELETE 操作涉及tuple 信息中
的旧值,对INSERT 为空。
TG_NAME
触发器名称。
TG_WHEN
触发器触发时机
(BEFORE/AFTER/INSTEA
D OF)。
TG_LEVEL
触发频率(ROW/STATEMENT)。
TG_OP
触发操作(INSERT/UPDATE/DELETE/TRU
NCATE)。

GBase 8c SQL 参考手册
南大通用数据技术股份有限公司
1071
变量名
变量含义
TG_RELID
触发器所在表OID。
TG_RELNAME
触发器所在表名(已废弃,现用TG_TABL
E_NAME 替代)。
TG_TABLE_NAME
触发器所在表名。
TG_TABLE_SCHEMA
触发器所在表的SCHEMA 信息。
TG_NARGS
触发器函数参数个数。
TG_ARGV[]
触发器函数参数列表。
示例
--创建源表及触发表
gbase=#CREATE TABLE test_trigger_src_tbl(id1 INT, id2 INT, id3 INT);
gbase=#CREATE TABLE test_trigger_des_tbl(id1 INT, id2 INT, id3 INT);
--创建触发器函数
gbase=#CREATE OR REPLACE FUNCTION tri_insert_func() RETURNS TRIGGER AS
$$
DECLARE
BEGIN
INSERT INTO test_trigger_des_tbl VALUES(NEW.id1, NEW.id2,
NEW.id3);
RETURN NEW;
END
$$ LANGUAGE PLPGSQL;
gbase=#CREATE OR REPLACE FUNCTION tri_update_func() RETURNS TRIGGER AS
$$
DECLARE

GBase 8c SQL 参考手册
南大通用数据技术股份有限公司
1072
BEGIN
UPDATE test_trigger_des_tbl SET id3 = NEW.id3 WHERE
id1=OLD.id1;
RETURN OLD;
END
$$ LANGUAGE PLPGSQL;
gbase=#CREATE OR REPLACE FUNCTION TRI_DELETE_FUNC() RETURNS TRIGGER AS
$$
DECLARE
BEGIN
DELETE FROM test_trigger_des_tbl WHERE id1=OLD.id1;
RETURN OLD;
END
$$ LANGUAGE PLPGSQL;
--创建INSERT 触发器
gbase=#CREATE TRIGGER insert_trigger
BEFORE INSERT ON test_trigger_src_tbl
FOR EACH ROW
EXECUTE PROCEDURE tri_insert_func();
--创建UPDATE 触发器
gbase=#CREATE TRIGGER update_trigger
AFTER UPDATE ON test_trigger_src_tbl
FOR EACH ROW
EXECUTE PROCEDURE tri_update_func();

GBase 8c SQL 参考手册
南大通用数据技术股份有限公司
1073
--创建DELETE 触发器
gbase=#CREATE TRIGGER delete_trigger
BEFORE DELETE ON test_trigger_src_tbl
FOR EACH ROW
EXECUTE PROCEDURE tri_delete_func();
--执行INSERT 触发事件并检查触发结果
gbase=#INSERT INTO test_trigger_src_tbl VALUES(100,200,300);
gbase=#SELECT * FROM test_trigger_src_tbl;
gbase=#SELECT * FROM test_trigger_des_tbl;
//查看触发操作是否生效。
--执行UPDATE 触发事件并检查触发结果
gbase=#UPDATE test_trigger_src_tbl SET id3=400 WHERE id1=100;
gbase=#SELECT * FROM test_trigger_src_tbl;
gbase=#SELECT * FROM test_trigger_des_tbl;
//查看触发操作是否生效
--执行DELETE 触发事件并检查触发结果
gbase=#DELETE FROM test_trigger_src_tbl WHERE id1=100;
gbase=#SELECT * FROM test_trigger_src_tbl;
gbase=#SELECT * FROM test_trigger_des_tbl;
//查看触发操作是否生效
--修改触发器
gbase=#ALTER TRIGGER delete_trigger ON test_trigger_src_tbl RENAME TO
delete_trigger_renamed;
--禁用insert_trigger 触发器
gbase=#ALTER TABLE test_trigger_src_tbl DISABLE TRIGGER insert_trigger;

GBase 8c SQL 参考手册
南大通用数据技术股份有限公司
1074
--禁用当前表上所有触发器
gbase=#ALTER TABLE test_trigger_src_tbl DISABLE TRIGGER ALL;
--删除触发器
gbase=#DROP TRIGGER insert_trigger ON test_trigger_src_tbl;
gbase=#DROP TRIGGER update_trigger ON test_trigger_src_tbl;
gbase=#DROP TRIGGER delete_trigger_renamed ON test_trigger_src_tbl;
相关命令
ALTER TRIGGER,DROP TRIGGER,ALTER TABLE

要从一致性检查获取最大好处并确保数据库空间备份的完整性,必须定期执行以下操作:

验证所有数据和数据库服务器开销信息是一致的。

检查消息日志是否在您验证一致性时有断言失败。

在您验证一致性后创建 0 级数据库空间备份。
以下主题描述了其中每个操作。
验证一致性
由于此检查需要一定时间并且此检查可能导致争用,因此请将此检查调度为在活动最少的
时候执行。必须在创建 0 级备份之前执行此检查。
运行下表中显示的命令以作为一致性检查的一部分。
表 1. 检查数据一致性
验证类型
命令

GBase 8s 管理员指南
南大通用数据技术股份有限公司
- 340 -
系统目录表
oncheck -cc
数据
oncheck -cD dbname
扩展数据块数
oncheck -ce
索引
oncheck -cI dbname
保留页
oncheck -cr
逻辑日志和保留页
oncheck -cR
元数据和智能大对象
oncheck -cs
您可以在数据库服务器处于联机方式时运行这些命令中的每个命令。有关每个命令如何在
检查对象时锁定它们以及哪些用户可以执行验证的信息,请参阅 《GBase 8s 管理员参
考》 中的 oncheck。
在大多数情况下,如果这些验证过程中的一个或多个过程检测到错误,那么解决方案是从
数据库空间备份复原数据库。但是,错误的来源也可能是硬件或操作系统。
验证系统目录表
要验证系统目录表,使用 oncheck -cc 命令。
每个数据库包含它本身的系统目录,该目录包含有关数据库表、列、索引、视图、约束、
存储过程和特权的信息。
如果当验证完成时显示警告,那么该警告的唯一用途是提醒您找不到特定类型的记录。这
些警告并非指示您的数据、系统目录或甚至数据库设计有任何问题。该警告仅指示不存在
任何表的同义词;即,系统目录在表 syssyntable 中不包含记录。 例如,如果您为没有为任
何表定义同义词的数据库验证系统目录表,那么可能显示以下警告:
WARNING: No syssyntable records found.
但是,如果您在验证系统目录表时接收到错误消息,那么情况就完全不同了。请立即
与 GBase 8s 技术支持联系。
验证数据页
要确认数据页,请使用 oncheck -cD 命令。
如果数据页验证检测到错误,请尝试从指定表中卸载数据,删除该表,重新创建该表,并
重新装入该数据。有关装入和卸载数据的信息,请参阅《GBase 8s 迁移指南》。如果该过
程不成功,请从存储空间备份执行数据复原。
验证扩展数据块
要确认每个数据库中的扩展数据块,请使用 oncheck -ce 命令。
扩展数据块一定不能重叠。如果该命令检测到错误,请从存储空间备份执行数据复原。

GBase 8s 管理员指南
南大通用数据技术股份有限公司
- 341 -
验证索引
如果索引已损坏,那么数据库服务器在查询中将无法使用索引。
可以通过使用 oncheck -cI 命令来验证数据库中每个表上的索引。
此外,调度程序任务 bad_index_alert 将查找已被服务器标记为已损坏的索引。此任务在每
晚运行。
对于此任务找到的每个已损坏索引,
都会在 sysadmin:ph_alert 表中建立一个条目。

如果索引已损坏,请将其删除,然后重新创建。
验证逻辑日志
要确认逻辑日志和保留页,请使用 oncheck -cR 命令。
验证保留页
要确认保留页,请使用 oncheck -cr 命令。
保留页是位于根数据库空间初始块开始处的页。这些页包含主数据库服务器开销信息。如
果该命令检测到错误,请从存储空间备份执行数据复原。
该命令可能会提供警告。在大多数情况下,这些警告让您注意的是您已经知道的情况。
验证元数据
对每个数据库运行 oncheck -cs 以验证数据库中所有智能大对象的元数据。
如有必要,
从数
据库空间备份复原数据。
监视数据不一致性
如果一致性检查代码在数据库服务器操作期间检测到不一致,将向数据库服务器消息日志
报告断言失败。(请参阅《GBase 8s 管理员参考》中的消息日志主题。)
消息日志和转储文件中的读断言失败
以下示例显示断言失败在消息日志中采用的格式。
Assert Failed: Short description of what failed
Who: Description of user/session/thread running at the time
Result: State of the affected database server entity
Action: What action the database server administrator should take
See Also: file(s) containing additional diagnostics
See Also: 行中包含以下一个或多个文件名:

af.xxx

shmem.xxx

gcore.xxx

gcore.xxx

/path name/core

GBase 8s 管理员指南
南大通用数据技术股份有限公司
- 342 -
在所有情况中,
xxx 对与单个线程的断言失败相关联的所有文件而言是一个公共的十六进制
数。文件 af.xxx、shmem.xxx 和 gcore.xxx 位于 ONCONFIG 参数 DUMPDIR 指定的目录
下。
文件 af.xxx 包含发送到消息日志的断言失败消息的副本以及当前相关结构和数据缓冲区的
内容。
仅当 ONCONFIG 参数 DUMPSHMEM 设置为 1 或 2 时,文件 shmem.xxx 才包含断言失
败时数据库服务器共享内存的完整副本。
仅限 UNIX:
在 UNIX™ 上,
仅当 ONCONFIG 参数 DUMPGCORE 设置为 1 且您的操作
系统支持 gcore 实用程序时,gcore.xxx 才包含数据库服务器虚拟进程(线程此时在其上运
行)的核心转储。仅当 ONCONFIG 参数 DUMPCORE 设置为 1 时,core 才包含数据库服
务器虚拟进程(线程此时在其上运行)的核心转储。core 文件的路径名是上次调用数据库
服务器的目录。
验证表和表空间数据
要验证表和表空间数据,请对数据库或表使用 oncheck -cD 命令。
大多数的一般断言失败消息后面均跟随其他信息,
其他信息通常包含检测到错误的表空间。
如果此检查确认存在不一致情况,请从表中卸载数据,删除该表,重新创建该表,并重新
装入该数据。否则,无需执行其他任何操作。
在许多情况中,数据库服务器当断言失败时会立即停止。但是,当失败看起来是特定于某
个表或较小实体时,数据库服务器会继续运行。
当断言是由于数据库服务器代表用户访问的数据页上的不一致而失败时,还会将错误发送
至应用程序进程。SQL 错误取决于正在进行的操作。但是,ISAM错误几乎始终是 -105 或
-172,如下所示:
-105 ISAM error: bad isam file format
-172 ISAM error: Unexpected internal error
有关消息的目标和内容的更多详细信息,请参阅《GBase 8s 管理员参考》中有关消息日志
消息的主题。
保留一致的 0 级备份
在您执行验证一致性中描述的检查而不出错之后,请创建 0 级备份。保留该存储空间备份
和所有后续的逻辑日志备份磁带,直至您完成下一次一致性检查。请在每次 0 级备份之前
执行一致性检查。如果不这样做,那么至少保留所有从存储空间备份恢复所需的磁带,该
备份是在验证数据库服务器一致之后立即创建的。

gbase_fetch_fields
............................
14