PostgreSQL实现了表继承,这对数据库设计者来说是一种有用的工具(SQL:1999及其后的版本定义了一种类型继承特性,但和这里介绍的继承有很大的不同)。
让我们从一个示例开始:假设我们要为城市建立一个数据模型。每个州有很多城市,但只有一个首府。我们希望能够快速检索任意特定州的首府城市。这可以通过创建两个表来实现:一个用于州首府,另一个用于非首府城市。然而,当我们想要查询某个城市的数据,而不关心它是不是首府时,会发生什么?继承特性将有助于解决这个问题。我们可以将capitals表定义为继承自cities表:
CREATE TABLE cities (
name text,
population float,
elevation int -- in feet
);
CREATE TABLE capitals (
state char(2)
) INHERITS (cities);
在这种情况下,capitals表继承了它的父表cities的所有列。州首府还有一个额外的列state用来表示它所属的州。
在PostgreSQL中,一个表可以从0个或者多个其他表继承,而对一个表的查询则可以引用一个表的所有行或者该表的所有行加上它所有的后代表。 默认情况是后一种行为。例如,下面的查询将查找所有高度高于500尺的城市的名称,包括州首府:
SELECT name, elevation
FROM cities
WHERE elevation > 500;
对于来自PostgreSQL教程(见Section 2.1)的示例数据,它将返回:
name | elevation -----------+----------- Las Vegas | 2174 Mariposa | 1953 Madison | 845
另一方面,下面的查询将找到所有高度超过 500 尺且不是州首府的城市:
SELECT name, elevation
FROM ONLY cities
WHERE elevation > 500;
name | elevation
-----------+-----------
Las Vegas | 2174
Mariposa | 1953
这里的ONLY关键词指示查询只被应用于cities上,而其他在继承层次中位于cities之下的其他表都不会被该查询涉及。很多我们已经讨论过的命令(如SELECT、UPDATE和DELETE)都支持ONLY关键词。
我们也可以在表名后写上一个*来显式地将后代表包括在查询范围内:
SELECT name, elevation
FROM cities*
WHERE elevation > 500;
写*并非必需,因为这种行为始终是默认的。不过,为了兼容那些允许修改默认行为的较旧版本,现在仍然支持这种语法。
在某些情况下,我们可能希望知道一个特定行来自于哪个表。每个表中的系统列tableoid可以告诉我们行来自于哪个表:
SELECT c.tableoid, c.name, c.elevation FROM cities c WHERE c.elevation > 500;
将会返回:
tableoid | name | elevation ----------+-----------+----------- 139793 | Las Vegas | 2174 139793 | Mariposa | 1953 139798 | Madison | 845
(如果重新生成这个结果,可能会得到不同的OID数字。)通过与pg_class进行连接可以看到实际的表名:
SELECT p.relname, c.name, c.elevation FROM cities c, pg_class p WHERE c.elevation > 500 AND c.tableoid = p.oid;
将会返回:
relname | name | elevation ----------+-----------+----------- cities | Las Vegas | 2174 cities | Mariposa | 1953 capitals | Madison | 845
得到同样效果的另一种方法,是使用regclass别名类型, 它会以符号形式打印表的 OID:
SELECT c.tableoid::regclass, c.name, c.elevation FROM cities c WHERE c.elevation > 500;
继承不会自动地将来自INSERT或COPY命令的数据传播到继承层次中的其他表中。在我们的示例中,下面的INSERT语句将会失败:
INSERT INTO cities (name, population, elevation, state)
VALUES ('Albany', NULL, NULL, 'NY');
我们也许会希望数据能以某种方式被路由到capitals表中,但这不会发生:INSERT总是向指定的表中插入。在某些情况下,可以通过使用一个规则(见Chapter 39)将插入动作重定向。但是这对上面的情况并没有帮助,因为cities表根本就不包含state列,因而这个命令会在触发规则之前就被拒绝。
父表上的所有检查约束和非空约束都将自动被它的后代所继承,除非显式地指定了NO INHERIT子句。其他类型的约束(唯一、主键和外键约束)则不会被继承。
一个表可以从超过一个的父表继承,在这种情况下它拥有父表们所定义的列的并集。任何定义在子表上的列也会被加入到其中。如果在这个集合中出现重名列,那么这些列将被“合并”,这样在子表中只会有一个这样的列。重名列能被合并的前提是这些列必须具有相同的数据类型,否则会导致错误。可继承的检查约束和非空约束会以类似的方式被合并。例如,如果合并成一个合并列的任一列定义被标记为非空,则该合并列会被标记为非空。如果检查约束的名称相同,则他们会被合并,但如果它们的条件不同则合并会失败。
表继承通常在创建子表时建立,即通过CREATE TABLE语句中的INHERITS子句。已经创建好的表也可以通过ALTER TABLE的INHERIT变体再增加一个新的父表关系。要这么做,新子表必须已经包含与父表同名且数据类型相同的列。子表还必须包含与父表相同的检查约束和检查表达式。类似地,也可以使用ALTER TABLE的NO INHERIT变体,从子表中移除一条继承链接。动态添加和移除继承链接可用于实现表分区(见Section 5.12)。
一种创建将来要用作子表的新表的方法,是在CREATE TABLE中使用LIKE子句。这样会创建一个与源表具有相同列的新表。如果源表上定义了任何CHECK约束,可以使用LIKE的INCLUDING CONSTRAINTS选项,让新表也包含与父表相同的约束。
当有任何一个子表存在时,父表不能被删除。当子表的列或者检查约束继承于父表时,它们也不能被删除或修改。如果希望移除一个表和它的所有后代,一种简单的方法是使用CASCADE选项删除父表(见Section 5.15)。
ALTER TABLE会把列的数据定义或检查约束上的任何变化沿着继承层次向下传播。同样,删除被其他表依赖的列只能使用CASCADE选项。对于同名列的合并与拒绝,ALTER TABLE遵循与CREATE TABLE相同的规则。
继承的查询仅在父表上执行访问权限检查。例如,在cities表上授予UPDATE权限也隐含着通过cities访问时在capitals表中更新行的权限。 这保留了数据(也)在父表中的样子。但是如果没有额外的授权,则不能直接更新capitals表。 以类似的方式,父表的行安全性策略(见Section 5.9)适用于继承查询期间来自于子表的行。 只有当子表在查询中被明确提到时,其策略(如果有)才会被应用,在那种情况下,附着在其父表上的任何策略都会被忽略。
外部表(见Section 5.13)也可以是继承层次 中的一部分,即可以作为父表也可以作为子表,就像常规表一样。如果 一个外部表是继承层次的一部分,那么任何不被该外部表支持的操作也 不被整个层次所支持。
注意,并非所有 SQL 命令都能作用于继承层次。用于数据查询、数据修改或模式修改的命令(例如SELECT、UPDATE、DELETE、大多数ALTER TABLE变体,但不包括INSERT或ALTER TABLE ... RENAME)通常默认包含子表,并支持使用ONLY记法将其排除。大多数用于数据库维护和调优的命令(例如REINDEX)只作用于独立的物理表,不支持沿继承层次递归。不过,VACUUM和ANALYZE默认会包含子表,并支持使用ONLY记法将其排除。各条命令的具体行为都记录在相应参考页中(SQL 命令)。
继承特性的一个严重限制是,索引(包括唯一约束)和外键约束只作用于单个表,而不作用于其继承子表。对于外键约束的引用端和被引用端,这一点都成立。因此,沿用上面的示例:
如果我们把cities.name声明为UNIQUE或PRIMARY KEY,这并不能阻止capitals表中出现与cities中城市同名的行。而且这些重复行默认还会出现在针对cities的查询结果中。事实上,默认情况下capitals根本没有唯一约束,因此它可以包含多行同名记录。你当然可以给capitals添加唯一约束,但这仍无法阻止相对于cities的重复。
相似地,如果我们指定cities.name REFERENCES某个其他表,该约束不会自动地传播到capitals。在此种情况下,我们可以变通地在capitals上手工创建一个相同的REFERENCES约束。
如果让另一个表的某列REFERENCES cities(name),那么该表可以包含城市名称,但不能包含首府名称。对于这种情况,并没有什么好的变通办法。
某些未为继承层次实现的功能,却已经为声明式分区实现了。因此,在决定使用旧式继承分区是否适合你的应用时,需要非常小心。
如果您发现文档中有不正确的内容、与您使用特定功能的经验不符或需要进一步说明,请使用此表单来报告文档问题。