2024 年 9 月 26 日: PostgreSQL 17 发布!
支持的版本:当前 (17) / 16 / 15 / 14 / 13 / 12
开发版本:devel

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 列列表中指定。

提交更正

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