ALTER TABLE添加复合主键


195

我有一张叫的桌子provider。我有三个列personplacething。可以有重复的人,重复的地方和重复的事物,但是永远不可能有重复的人与事物的组合。

我如何使用上述三列更改表以在MySQL中为此表添加复合主键?

Answers:


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

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

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

17
@ David542不,不是-您只能有1个主键。
Adrian Cornish 2012年

35
@David:这是由多个字段组成的单个主键,也称为复合键。
Marc B

3
@ David542当然可以-这是一个由3个字段组成的复合主键。这三个字段的组合必须唯一。
Adrian Cornish 2012年

2
感谢您的发帖-真的让我摆脱了与ui的斗争
plditallo 2013年

1
SO中最珍贵的答案之一:)
alwbtc

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).

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

另外,使用组合键时,顺序很重要。这些

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的文档在这里获取更多信息。


@All-您是否可以共享同等的JPA?
Pra_A

17

您可能只需要UNIQUE CONSTRAINT。特别是如果您已经具有代理密钥。(一个已经存在的替代键的示例将是一个单列,即AUTO_INCREMENT)

以下是唯一约束的sql代码

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

谢谢,约束是我想要的,我不知道在最初的帖子中要问什么。感谢您将此添加到线程中。
ZaneDarken 2014年

我通常使用代理键……然后添加一个唯一约束。这样,...如果“唯一性”改变了,调整约束就不算什么大麻烦了,而把主键搞砸了。并且,如果您的子表中有外键引用此表,则只需FK代理键,而不是全部3列。–
granadaCoder 2014年


1

正如@GranadaCoder提供的那样,使用COMPOSITE UNIQUE KEY绝对是更好的选择,但是有些棘手的示例:

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


0

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

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.