ALTER TABLE添加复合主键

时间:2012-01-14 01:20:27

标签: mysql sql primary-key alter-table composite-primary-key

我有一张名为provider的表格。我有三列名为personplacething。可能有重复的人,重复的地方和重复的东西,但永远不会有重复的人物 - 地方 - 事物组合。

我如何使用这三列在ALTER TABLE中为此表添加复合主键?

6 个答案:

答案 0 :(得分:391)

ALTER TABLE provider ADD PRIMARY KEY(person,place,thing);

如果主键已经存在,那么你想要这样做

ALTER TABLE provider DROP PRIMARY KEY, ADD PRIMARY KEY(person, place, thing);

答案 1 :(得分:19)

@Adrian Cornish的回答是正确的。但是,删除现有主键还有另一个警告。如果该主键被另一个表用作外键,则在尝试删除它时会出错。在某些版本的mysql中,错误消息格式错误(从5.5.17开始,此错误消息仍然是

alter table parent  drop column id;
ERROR 1025 (HY000): Error on rename of
'./test/#sql-a04_b' to './test/parent' (errno: 150).

如果要删除另一个表引用的主键,则必须先将该外键放在该另一个表中。如果在重新创建主键后仍需要该外键,则可以重新创建该外键。

此外,使用复合键时,顺序很重要。 这些

1) ALTER TABLE provider ADD PRIMARY KEY(person,place,thing);
and
2) ALTER TABLE provider ADD PRIMARY KEY(person,thing,place);

不是一回事。 它们都在这三个字段中强制实现唯一性,但从索引的角度来看,存在差异。字段从左到右编制索引。 例如,请考虑以下查询:

A) SELECT person, place, thing FROM provider WHERE person = 'foo' AND thing = 'bar';
B) SELECT person, place, thing FROM provider WHERE person = 'foo' AND place = 'baz';
C) SELECT person, place, thing FROM provider WHERE person = 'foo' AND place = 'baz' AND thing = 'bar';
D) SELECT person, place, thing FROM provider WHERE place = 'baz' AND thing = 'bar';

B可以使用ALTER语句1中的主键索引 A可以使用ALTER语句2中的主键索引 C可以使用索引
D不能使用索引

A使用索引2中的前两个字段作为部分索引。 A不能使用索引1,因为它不知道索引的中间位置部分。尽管如此,它仍然可以使用部分索引。

D不能使用任何索引,因为它不了解人。

有关详细信息,请参阅mysql文档here

答案 2 :(得分:15)

您可能只想要一个独特的约束。特别是如果你已经有一个代理键。 (已存在的代理键的示例是单个列,即AUTO_INCREMENT)

以下是Unique Constraint的SQL代码

ALTER TABLE `MyDatabase`.`Provider`
    ADD CONSTRAINT CK_Per_Place_Thing_Unique UNIQUE (person,place,thing)
;

答案 3 :(得分:3)

alter table table_name add primary key (col_name1, col_name2);

答案 4 :(得分:1)

使用COMPOSITE UNIQUE KEY肯定会更好,正如@GranadaCoder提供的那样,虽然有点棘手:

ALTER IGNORE TABLE table_name ADD UNIQUES INDEX idx_name(some_id, another_id, one_more_id);

答案 5 :(得分:0)

ALTER TABLE table_name DROP PRIMARY KEY,ADD PRIMARY KEY (col_name1, col_name2);