受支持版本: 当前版本 (18) / 17 / 16 / 15 / 14
开发版本: devel

F.39. spi — 服务器编程接口特性/示例 #

spi 模块提供了若干使用 服务器编程接口SPI)和触发器的实用示例。 虽然这些函数本身也有一定的独立用途,但把它们作为可按自身需求修改的示例来看更有价值。 这些函数足够通用,可以用于任何表,但在创建触发器时必须指定表名和字段名 (如下所述)。

下文描述的每组函数都作为一个可单独安装的扩展提供。

F.39.1. refint — 用于实现引用完整性的函数 #

check_primary_key()check_foreign_key() 用于检查外键约束。 (当然,这项功能早已被内置外键机制取代,不过该模块作为示例仍然很有用。)

check_primary_key() 检查引用表。使用时,在引用其他表的表上 创建一个使用该函数的 AFTER INSERT OR UPDATE 触发器。 触发器参数依次为:引用表中构成外键的列名、被引用表名,以及被引用表中构成 主键/唯一键的列名。 若要处理多个外键,请为每个引用分别创建一个触发器。

check_foreign_key() 检查被引用表。使用时,在被其他表引用的表上 创建一个使用该函数的 AFTER DELETE OR UPDATE 触发器。 触发器参数依次为:该函数需要检查的引用表数量、发现引用键时采取的动作 (cascade 表示删除引用行, restrict 表示若存在引用键则中止事务, setnull 表示将引用键字段设为空)、 触发表中构成主键/唯一键的列名,然后是引用表名和列名 (按第一个参数指定的引用表数量重复提供)。请注意,主键/唯一键列应标记为 NOT NULL,并具有唯一索引。

请注意,如果这些触发器是从另一个 BEFORE 触发器中执行的, 它们可能会意外失败。例如,如果用户插入 row1,然后该 BEFORE 触发器插入 row2 并调用使用 check_foreign_key() 的触发器, 那么 check_foreign_key() 函数将看不到 row1,因而失败。

refint.example 中有示例。

F.39.2. autoinc — 用于字段自动递增的函数 #

autoinc() 是一个将序列的下一个值存入整数字段的触发器。 这与内置的 serial 列 功能有一些重叠,但并不相同。 只有当该字段的值最初为零或空值时 (也就是在插入或更新该行的 SQL 语句动作执行之后),触发器才会替换该字段值。 此外,如果序列的下一个值为零,则会再次调用 nextval(), 以取得一个非零值。

使用时,创建一个使用该函数的 BEFORE INSERT (或者可选的 BEFORE INSERT OR UPDATE)触发器。 指定两个触发器参数:要修改的整型列名,以及提供这些值的序列对象名。 (实际上,如果你希望更新多个自动递增列,也可以指定任意多组这样的名称对。)

autoinc.example 中有一个示例。

F.39.3. insert_username — 用于跟踪谁修改了表的函数 #

insert_username() 是一个将当前用户名存入文本字段的触发器。 它有助于跟踪表中某一特定行最后一次是由谁修改的。

使用时,创建一个使用该函数的 BEFORE INSERT 和/或 UPDATE 触发器。指定一个触发器参数:要修改的文本列名。

insert_username.example 中有一个示例。

F.39.4. moddatetime — 用于跟踪最后修改时间的函数 #

moddatetime() 是一个将当前时间存入 timestamp 字段的触发器。 它有助于跟踪表中某一特定行最后一次被修改的时间。

使用时,创建一个使用该函数的 BEFORE UPDATE 触发器。 指定一个触发器参数:要修改的列名。该列必须是 timestamptimestamp with time zone 类型。

moddatetime.example 中有一个示例。

提交更正

如果您发现文档中有不正确的内容、与您使用特定功能的经验不符或需要进一步说明,请使用此表单来报告文档问题。