支持的版本: 当前 (17) / 16 / 15 / 14 / 13
开发版本: devel
不支持的版本: 12

5.4. 生成列 #

生成列是一种特殊的列,它总是由其他列计算而来。因此,对于列来说,它就像视图对于表一样。生成列有两种类型:存储的和虚拟的。存储的生成列在写入(插入或更新)时计算,并像普通列一样占用存储空间。虚拟生成列不占用存储空间,而是在读取时计算。因此,虚拟生成列类似于视图,而存储的生成列类似于物化视图(只是它总是自动更新)。PostgreSQL 目前仅实现存储的生成列。

要创建生成列,请在 CREATE TABLE 中使用 GENERATED ALWAYS AS 子句,例如

CREATE TABLE people (
    ...,
    height_cm numeric,
    height_in numeric GENERATED ALWAYS AS (height_cm / 2.54) STORED
);

必须指定关键字 STORED 以选择存储类型的生成列。有关更多详细信息,请参见 CREATE TABLE

不能直接写入生成列。在 INSERTUPDATE 命令中,不能为生成列指定值,但可以指定关键字 DEFAULT

考虑具有默认值的列和生成列之间的差异。当第一次插入行且未提供其他值时,会评估列默认值一次;生成列会在行更改时更新,并且不能被覆盖。列默认值可能不会引用表的其他列;而生成表达式通常会这样做。列默认值可以使用易变函数,例如 random() 或引用当前时间的函数;这在生成列中是不允许的。

以下几个限制适用于生成列的定义和涉及生成列的表

  • 生成表达式只能使用不可变函数,并且不能使用子查询或以任何方式引用当前行之外的任何内容。

  • 生成表达式不能引用另一个生成列。

  • 生成表达式不能引用系统列,除了 tableoid

  • 生成列不能具有列默认值或标识定义。

  • 生成列不能是分区键的一部分。

  • 外部表可以有生成列。有关详细信息,请参见 CREATE FOREIGN TABLE

  • 对于继承和分区

    • 如果父列是生成列,则其子列也必须是生成列;但是,子列可以具有不同的生成表达式。在插入或更新行时实际应用的生成表达式是与该行物理所在的表关联的表达式。(这与列默认值的行为不同:对于列默认值,适用于查询中命名的表关联的默认值。)

    • 如果父列不是生成列,则其子列也不能是生成列。

    • 对于继承的表,如果在 CREATE TABLE ... INHERITS 中编写子列定义时没有 GENERATED 子句,则其 GENERATED 子句将自动从父级复制。ALTER TABLE ... INHERIT 将坚持父列和子列在生成状态方面已经匹配,但它不会要求它们的生成表达式匹配。

    • 对于分区表也类似,如果在 CREATE TABLE ... PARTITION OF 中编写子列定义时没有 GENERATED 子句,则其 GENERATED 子句将自动从父级复制。ALTER TABLE ... ATTACH PARTITION 将坚持父列和子列在生成状态方面已经匹配,但它不会要求它们的生成表达式匹配。

    • 在多重继承的情况下,如果一个父列是生成列,则所有父列都必须是生成列。如果它们不都具有相同的生成表达式,则必须显式指定子列所需的表达式。

以下其他注意事项适用于生成列的使用。

  • 生成列维护与它们的底层基本列不同的访问权限。因此,可以安排让特定角色可以从生成列读取,但不能从底层基本列读取。

  • 从概念上讲,生成列在 BEFORE 触发器运行后更新。因此,在 BEFORE 触发器中对基本列所做的更改将反映在生成列中。但反之,不允许在 BEFORE 触发器中访问生成列。

  • 生成列会被跳过逻辑复制,并且不能在 CREATE PUBLICATION 列列表中指定。

提交更正

如果您在文档中发现任何不正确、与您使用特定功能的经验不符或需要进一步澄清的内容,请使用此表格报告文档问题。