ALTER TABLE添加复合主键

Dav*_*542 186 mysql sql alter-table primary-key composite-primary-key

我有一张叫做的桌子provider.我有三个列person,place,thing.可能存在重复的人,重复的地方和重复的事物,但是永远不会存在重复的人物 - 事物 - 组合.

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

Adr*_*ish 404

ALTER TABLE provider ADD PRIMARY KEY(person,place,thing);
Run Code Online (Sandbox Code Playgroud)

如果主键已存在,那么您希望这样做

ALTER TABLE provider DROP PRIMARY KEY, ADD PRIMARY KEY(person, place, thing);
Run Code Online (Sandbox Code Playgroud)

  • @David:它是由多个字段组成的单个主键,也称为复合键. (34认同)
  • @ David542不,它没有 - 你只能有一个主键. (17认同)
  • @David542当然你可以 - 这是一个由3个字段组成的复合主键.3个字段的组合必须是唯一的. (3认同)
  • 这增加了三个独特的pks,而不是一个复合pk. (2认同)
  • 感谢发布 - 真的让我没有与ui作斗争 (2认同)

cs_*_*nus 21

@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).
Run Code Online (Sandbox Code Playgroud)

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

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

1) ALTER TABLE provider ADD PRIMARY KEY(person,place,thing);
and
2) ALTER TABLE provider ADD PRIMARY KEY(person,thing,place);
Run Code Online (Sandbox Code Playgroud)

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

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';
Run Code Online (Sandbox Code Playgroud)

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

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

D不能使用任何索引,因为它不知道人.

有关更多信息,请参阅此处的mysql文档.


gra*_*der 15

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

下面是Unique Constraint的sql代码

ALTER TABLE `MyDatabase`.`Provider`
    ADD CONSTRAINT CK_Per_Place_Thing_Unique UNIQUE (person,place,thing)
;
Run Code Online (Sandbox Code Playgroud)