首页 文章

ALTER TABLE添加复合主键

提问于
浏览
166

我有一个名为 provider 的表 . 我有三列名为 personplacething . 可能存在重复的人,重复的地方和重复的事物,但是永远不会存在重复的人物 - 事物 - 组合 .

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

6 回答

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

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

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

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

  • 1

    您可能只想要一个独特的约束 . 特别是如果你已经有一个代理键 . (例如:一个AUTO_INCREMENT)

    ALTER TABLE `MyDatabase`.`Provider`
        ADD CONSTRAINT CK_Per_Place_Thing_Unique UNIQUE (person,place,thing)
    ;
    
  • 14
    alter table table_name add primary key (col_name1, col_name2);
    
  • 19

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

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

  • 373

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

相关问题