这一节描述了一个触发器函数的接口的低层细节。只有用 C 编写触发器函数时才需要这些信息。如果你使用一种更高层的语言,那么这些细节就不需要你来处理。在大部分情况下,你应该优先考虑使用一种过程语言。每一种过程语言的文档阐述了如何使用那种语言编写一个触发器。
触发器函数必须使用“版本 1”函数管理器接口。
当一个函数被触发器管理器调用时,不会给它传递任何常规的参数,但是会有一个“context”指针传递给它,该指针指向一个TriggerData
结构。C 函数可以通过执行一个宏来检查它们是否是从触发器管理器被调用:
CALLED_AS_TRIGGER(fcinfo)
它会展开成为:
((fcinfo)->context != NULL && IsA((fcinfo)->context, TriggerData))
如果这返回真,那么将fcinfo->context
造型成类型TriggerData *
并且利用所指向的TriggerData
结构就是安全的。该函数不能修改该TriggerData
结构或者它指向的任何数据。
struct TriggerData
被定义在commands/trigger.h
中:
typedef struct TriggerData
{
NodeTag type;
TriggerEvent tg_event;
Relation tg_relation;
HeapTuple tg_trigtuple;
HeapTuple tg_newtuple;
Trigger *tg_trigger;
TupleTableSlot *tg_trigslot;
TupleTableSlot *tg_newslot;
Tuplestorestate *tg_oldtable;
Tuplestorestate *tg_newtable;
const Bitmapset *tg_updatedcols;
} TriggerData;
其中的成员被定义如下:
type
总是T_TriggerData
.
tg_event
描述该函数是为什么事件被调用的。你可以使用下列宏来检查tg_event
:
TRIGGER_FIRED_BEFORE(tg_event)
如果该触发器在操作前被引发则返回真。
TRIGGER_FIRED_AFTER(tg_event)
如果该触发器在操作后被引发则返回真。
TRIGGER_FIRED_INSTEAD(tg_event)
如果该触发器被引发替代操作则返回真。
TRIGGER_FIRED_FOR_ROW(tg_event)
如果该触发器为一个行级事件而引发则返回真。
TRIGGER_FIRED_FOR_STATEMENT(tg_event)
如果该触发器为一个语句级事件而引发则返回真。
TRIGGER_FIRED_BY_INSERT(tg_event)
如果该触发器由一个INSERT
命令引发则返回真。
TRIGGER_FIRED_BY_UPDATE(tg_event)
如果该触发器由一个UPDATE
命令引发则返回真。
TRIGGER_FIRED_BY_DELETE(tg_event)
如果该触发器由一个DELETE
命令引发则返回真。
TRIGGER_FIRED_BY_TRUNCATE(tg_event)
如果该触发器由一个TRUNCATE
命令引发则返回真。
tg_relation
一个结构指针,该结构描述该触发器为其引发的关系。关于这个结构的细节请参考utils/rel.h
。最有趣的东西是tg_relation->rd_att
(该关系元组的描述符)和tg_relation->rd_rel->relname
(关系名称,该类型不是char*
而是
NameData
。如果你需要该名称的一个拷贝,可使用SPI_getrelname(tg_relation)
来得到一个char*
)。
tg_trigtuple
一个该触发器为其引发的行的指针。这是被插入、更新或删除的行。如果这个触发器是为一个INSERT
或DELETE
而引发,在你不想把该行替换成另一行(在INSERT
的情况中)或不想跳过该操作时你应该从该函数中返回它。 对于外部表上的触发器,此中的系统列值未被指定。
tg_newtuple
如果该触发器为一个UPDATE
而引发,则是一个指向该行新版本的指针。如果是为一个INSERT
或DELETE
而引发,则是NULL
。如果事件是一个UPDATE
并且你并不想用一个不同的行替换这个行或者不想跳过该操作时,你必须从函数中返回它。对于外部表上的触发器,此中的系统列值未被指定。
tg_trigger
一个指向类型为Trigger
的结构的指针,定义在utils/reltrigger.h
中:
typedef struct Trigger
{
Oid tgoid;
char *tgname;
Oid tgfoid;
int16 tgtype;
char tgenabled;
bool tgisinternal;
Oid tgconstrrelid;
Oid tgconstrindid;
Oid tgconstraint;
bool tgdeferrable;
bool tginitdeferred;
int16 tgnargs;
int16 tgnattr;
int16 *tgattr;
char **tgargs;
char *tgqual;
char *tgoldtable;
char *tgnewtable;
} Trigger;
其中tgname
是该触发器的名称,tgnargs
是tgargs
中参数的数量,而tgargs
是一个指向CREATE TRIGGER
语句中指定的参数的指针数组。其他成员只用于内部用途。
tg_trigtuplebuf
包含tg_trigtuple
的插槽。或者一个NULL
指针,如果没有这样的元组的话。
tg_newtuplebuf
包含tg_trigtuple
的插槽。或者一个NULL
指针,如果没有这样的元组的话。
tg_oldtable
一个指向Tuplestorestate
类型的结构的指针,该结构包含格式由tg_relation
指定的零行或者多行。如果没有OLD TABLE
传递关系,则为NULL
指针。
tg_newtable
一个指向Tuplestorestate
类型的结构的指针,该结构包含格式由tg_relation
指定的零行或者多行。如果没有NEW TABLE
传递关系,则为NULL
指针。
tg_updatedcols
对于 UPDATE
触发器,位图集指示由触发命令更新的列。 通用触发器函数可以使用它来优化操作,而不必处理未更改的列。
例如,要确定属性编号为 attnum
(基于 1)的列是否是此位图集的成员,请调用 bms_is_member(attnum -
FirstLowInvalidHeapAttributeNumber,
trigdata->tg_updatedcols))
。
对于UPDATE
触发器以外的触发器,这将是NULL
。
为了允许通过SPI发出的查询引用传递表,请参考SPI_register_trigger_data。
一个触发器函数必须返回一个HeapTuple
指针或一个NULL
指针(不是一个 SQL 空值,也就是不会设置isNull
为真)。如果你不希望修改正在被操作的行,要小心地根据情况返回tg_trigtuple
或
tg_newtuple
。
42.1.1. 使用PL/pgSQL的优点42.1.2. 支持的参数和结果数据类型 PL/pgSQL是一种用于PostgreSQL数据库系统的可载入的过程语言。PL/...
下面的命令可以用来从 PL/Tcl 函数体中访问数据库:spi_exec ?-count n? ?-array name? command ?loop-body?执行一个以字符串给...
PL/Python 同时支持 Python 2 和 Python 3 两种语言变体(PostgreSQL 安装指导可能包含了所支持的 Python 次版本的更精确的信息...
plpy模块也提供了函数plpy.debug(msg, **kwargs)plpy.log(msg, **kwargs)plpy.info(msg, **kwargs)plpy.notice(msg, **kwargs)pl...
Image align 属性 Image 对象定义和用法align 属性可设置或者返回图像的 align 属性值。align 属性指定了与内联内容的对齐方式。...
Style orphans 属性 Style 对象定义和用法orphans 属性设置或返回一个元素必须在页面底部的可见行的最小数量(用于打印或打印预...
Style pageBreakInside 属性 Style 对象定义和用法pageBreakInside 属性设置或返回元素内的分页行为(用于打印或打印预览)。注...
Style minWidth 属性 Style 对象定义和用法minWidth 属性设置或返回元素的最小宽度。minWidth 属性只在块级元素或绝对/固定位置...
Table rules 属性 Table 对象定义和用法rules 属性可设置或返回表格的内部边线。语法设置 rules 属性:tableObject.rules="none|...
Style width 属性 Style 对象定义和用法width 属性设置或返回元素的宽度。width 属性只在块级元素或绝对/固定位置的元素发挥作用...