本文介绍了如何修改表。
简介
当我们已经创建了一个表并意识到犯了一个错误或者应用需求发生改变时,我们可以移除表并重新创建它。但如果表中已经被填充数据或者被其他数据库对象引用(例如有一个外键约束),这种做法就显得很不方便。因此,本数据库提供了一族命令来对已有的表进行修改。注意这和修改表中所包含的数据是不同的,这里要做的是对表的定义或者说结构进行修改。
利用这些命令,我们可以进行以下操作:
增加列
移除列
增加约束
移除约束
修改默认值
修改列数据类型
重命名列
重命名表
增加列
执行以下命令,增加一个列:
ALTER TABLE products ADD COLUMN description text;
新列将被默认值所填充(如果没有指定DEFAULT
子句,则会填充空值)。
从本数据库11开始,添加一个具有常量默认值的列不再意味着在执行ALTER TABLE
语句时需要更新表的每一行。 相反,默认值将在下次访问该行时返回,并在表被重写时应用,从而使得ALTER TABLE
即使在大表上也非常快。
但是,如果默认值是可变的(例如clock_timestamp()
),则每一行需要被ALTER TABLE
被执行时计算的值更新。 为避免潜在的长时间的更新操作,特别是如果你想要用大多数非默认值填充列,那么最好添加没有默认值的列,再用 UPDATE
插入正确的值,然后按照下面所述添加任何期望的默认值。
也可以同时为列定义约束,语法:
ALTER TABLE products ADD COLUMN description text CHECK (description <> '');
事实上CREATE TABLE
中关于一列的描述都可以应用在这里。记住不管怎样,默认值必须满足给定的约束,否则ADD
将会失败。也可以先将新列正确地填充好,然后再增加约束(见后文)。
移除列
执行以下命令,移除一个列:
ALTER TABLE products DROP COLUMN description;
列中的数据将会消失。涉及到该列的表约束也会被移除。然而,如果该列被另一个表的外键所引用,本数据库不会安静地移除该约束。我们可以通过增加CASCADE
来授权移除任何依赖于被删除列的所有东西:
ALTER TABLE products DROP COLUMN description CASCADE;
增加约束
为了增加一个约束,可以使用表约束的语法,例如:
ALTER TABLE products ADD CHECK (name <> '');
ALTER TABLE products ADD CONSTRAINT some_name UNIQUE (product_no);
ALTER TABLE products ADD FOREIGN KEY (product_group_id) REFERENCES product_groups;
要增加一个不能写成表约束的非空约束,可使用语法:
ALTER TABLE products ALTER COLUMN product_no SET NOT NULL;
该约束会立即被检查,所以表中的数据必须在约束被增加之前就已经符合约束。
移除约束
为了移除一个约束首先需要知道它的名称。如果在创建时已经给它指定了名称,那么事情就变得很容易。否则约束的名称是由系统生成的,我们必须先找出这个名称。psql的命令\d ``表名
将会对此有所帮助,其他接口也会提供方法来查看表的细节。因此命令如下:
ALTER TABLE products DROP CONSTRAINT some_name;
如果处理的是自动生成的约束名称,如$2
,别忘了用双引号使它变成一个合法的标识符。
和移除一个列相似,如果需要移除一个被某些别的东西依赖的约束,也需要加上CASCADE
。一个例子是一个外键约束依赖于被引用列上的一个唯一或者主键约束。
这对除了非空约束之外的所有约束类型都一样有效。为了移除一个非空约束可以用:
ALTER TABLE products ALTER COLUMN product_no DROP NOT NULL;
更改列的默认值
执行以下命令,为一个列设置一个新默认值:
ALTER TABLE products ALTER COLUMN price SET DEFAULT 7.77;
这不会影响任何表中已经存在的行,它只是为未来的INSERT
命令改变了默认值。
执行以下命令,移除任何默认值:
ALTER TABLE products ALTER COLUMN price DROP DEFAULT;
这等同于将默认值设置为空值。相应的,试图删除一个未被定义的默认值并不会引发错误,因为默认值已经被隐式地设置为空值。
修改列的数据类型
为了将一个列转换为一种不同的数据类型,使用如下命令:
ALTER TABLE products ALTER COLUMN price TYPE numeric(10,2);
只有当列中的每一个项都能通过一个隐式造型转换为新的类型时该操作才能成功。如果需要一种更复杂的转换,应该加上一个USING
子句来指定应该如何把旧值转换为新值。
本数据库将尝试把列的默认值转换为新类型,其他涉及到该列的任何约束也是一样。但是这些转换可能失败或者产生奇特的结果。因此最好在修改类型之前先删除该列上所有的约束,然后在修改完类型后重新加上相应修改过的约束。
重命名列
执行以下命令,重命名一个列:
ALTER TABLE products RENAME COLUMN product_no TO product_number;
重命名表
执行以下命令,重命名一个表:
ALTER TABLE products RENAME TO items;