返回介绍

I. 教程

II. SQL 语言

III. 服务器管理

IV. 客户端接口

V. 服务器端编程

VI. 参考手册

VII. 内部

VIII. 附录

38.6. PL/Tcl 里的触发器过程

发布于 2019-09-30 03:09:33 字数 2564 浏览 1173 评论 0 收藏 0

触发器过程可以用 PL/Tcl 写。PostgreSQL 要求当做触发器调用的过程必需声明为没有参数并且返回类型为 trigger 的函数。

触发器管理器传递给过程体的信息是通过下面变量传递的:

$TG_name

CREATE TRIGGER 语句里的触发器名称。

$TG_relid

导致触发器被调用的表的对象 ID 。

$TG_table_name

导致触发器被调用的表的名字。

$TG_table_schema

导致触发器被调用的表的模式。

$TG_relatts

以一个空表元素为前导的表中字段名称的 Tcl 列表。所以用 Tcl 命令 lsearch 在列表里查找元素名称时,返回的从 1 开始计数的正整数,与 PostgreSQL 里字段编号的传统一样。已经被删除掉的字段位置的空的列表元素仍然会出现,这样,属性编号与字段的对应就是正确的。

$TG_when

由触发器调用事件决定的字符串 BEFOREAFTER

$TG_level

由触发器调用事件决定的字符串 ROWSTATEMENT

$TG_op

由触发器调用事件决定的字符串 INSERT, UPDATE, DELETE

$NEW

一个关联数组,包含 INSERTUPDATE 动作的新表行,如果是 DELETE 则为空。该数组是用字段名做索引的。那些为空的字段不会在数组中出现。

$OLD

一个关联数组,包含 UPDATEDELETE 动作的新表行,如果是 INSERT 则为空。该数组是用字段名做索引的。那些为空的字段不会在数组中出现。

$args

如同在 CREATE TRIGGER 语句里给出的参数一样的 Tcl 参数表。这些参数在过程体里可以通过 $1 ... $n 来访问。

触发器过程返回的值是字符串 OKSKIP 之一,或者一个像 array get Tcl 命令返回的数组那样的东西。如果返回值是 OK ,触发触发器的操作(INSERT/UPDATE/DELETE)将会正常进行。SKIP 告诉触发器管理器不声不响地忽略该行的操作。如果返回一个数组,那么它告诉 PL/Tcl 返回一个修改后的行给触发器管理器,该行将代替在 $NEW(只在 INSERTUPDATE 中起作用)中给出的行。当然,这些只有在触发器是 BEFOREFOR EACH ROW 时才有意义;否则返回值将被忽略。

下面是一个小的触发器过程的例子,它强制表内的一个整数值对行的更新次数进行跟踪。对插入的新行,该值初始化为 0 并且在每次更新操作中加一。

CREATE FUNCTION trigfunc_modcount() RETURNS trigger AS $$
    switch $TG_op {
        INSERT {
            set NEW($1) 0
        }
        UPDATE {
            set NEW($1) $OLD($1)
            incr NEW($1)
        }
        default {
            return OK
        }
    }
    return [array get NEW]
$$ LANGUAGE pltcl;

CREATE TABLE mytab (num integer, description text, modcnt integer);

CREATE TRIGGER trig_mytab_modcount BEFORE INSERT OR UPDATE ON mytab
    FOR EACH ROW EXECUTE PROCEDURE trigfunc_modcount('modcnt');

请注意触发器过程本身并不知道字段名字;那些是从触发器参数中提供的。这样就让可以将触发器过程复用于不同的表。

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文