5

我在 MySQL InnoDB 中有一个这样创建的表:

CREATE TABLE `users` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT,
  `type` enum('MANUAL','FACEBOOK') NOT NULL DEFAULT 'MANUAL',
  `role` enum('COOK','HOST','ALL') NOT NULL DEFAULT 'ALL',
  `about_me` varchar(1000) DEFAULT NULL,
  `food_preferences` varchar(1000) DEFAULT NULL,
  `cooking_experience` varchar(1000) DEFAULT NULL,
  `with_friend` bit(1) DEFAULT b'0',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8

接下来我尝试使用这样的语句添加一个表(在创建表时没有添加外键,因为有问题):

CREATE TABLE `messages` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT,
  `from` bigint(20) NOT NULL,
  `to` bigint(20) NOT NULL,
  `content` varchar(10000) NOT NULL,
  `timestamp_sent` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
  `timestamp_read` timestamp NULL DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8

创建表后,我需要最后添加外键'from''to'引用'users'.'id'字段的字段:

ALTER TABLE `messages` 
  ADD CONSTRAINT `messages_users_fk`
  FOREIGN KEY (`from` , `to` )
  REFERENCES `users` (`id` , `id` )
  ON DELETE SET NULL
  ON UPDATE CASCADE
, ADD INDEX `messages_users_fk_idx` (`from` ASC, `to` ASC)

我得到的错误是:

错误:错误 1822:无法添加外键约束。引用表 'users' 中的约束 'messages_users_fk' 缺少索引。

但是“用户”表有一个PRIMARY索引'id'...

还尝试采取更小的步骤并仅为'from'字段添加外键:

ALTER TABLE `messages` 
  ADD CONSTRAINT `messages_users_fk`
  FOREIGN KEY (`from` )
  REFERENCES `users` (`id` )
  ON DELETE SET NULL
  ON UPDATE CASCADE
, ADD INDEX `messages_users_fk_idx` (`from` ASC) ;

错误略有不同:

错误:错误 1825:无法在表“消息”上添加外键约束。FOREIGN KEY 约束“cook4food/messages_users_fk”中的选项不正确。

bigint(20) NOT NULL字段的类型与其他 StackOverflow 线程中建议的问题原因相同 ( )。我的表没有分区(MySQL 手册指出这是在 InnoDB 中具有外键约束的限制)。该'messages'表当前不存储任何行,因此存储的数据不会成为任何问题。我被卡住了,请帮忙。

4

3 回答 3

15

如果您具有适当的权限,则可以发出以下查询:

SHOW ENGINE innodb STATUS

...这将告诉您(以及其他一些信息)错误的确切细节:

------------------------
LATEST FOREIGN KEY ERROR
------------------------
130311 13:30:06 Error in foreign key constraint of table test/#sql-71c_6:

  FOREIGN KEY (`from` , `to` )
  REFERENCES `users` (`id` , `id` )
  ON DELETE SET NULL
  ON UPDATE CASCADE
, ADD INDEX `messages_users_fk_idx` (`from` ASC, `to` ASC):
You have defined a SET NULL condition though some of the
columns are defined as NOT NULL.

在删除父记录时,您不能将子记录设为 NULL,因为这些列是NOT NULL

  `from` bigint(20) NOT NULL,
  `to` bigint(20) NOT NULL,

编辑:此外,我看不到单个复合键的用途。我认为您需要两个单键。

于 2013-03-11T12:34:13.067 回答
0

正如我们所看到的,

from当子列不为空时,您不能在子记录中定义 SET NULL to

因此,如果您尝试使用以下 2 条语句创建外键,则messages表中应该有外键。

1>

ALTER TABLE `messages`  ADD CONSTRAINT `messages_users_from_fk`   FOREIGN KEY (`from` )   REFERENCES `users` (`id` )  ON DELETE CASCADE ON UPDATE CASCADE, ADD INDEX `messages_users_fk_idx` (`from` ASC) ;

2>

ALTER TABLE `messages`  ADD CONSTRAINT `messages_users_to_fk`   FOREIGN KEY (`to` )   REFERENCES `users` (`id` )  ON DELETE CASCADE ON UPDATE CASCADE, ADD INDEX `messages_users_fk_to_idx` (`to` ASC) ;

希望这些陈述对您有所帮助。

于 2013-03-11T12:55:25.533 回答
0

REFERENCES用户(id ,id)

您正在尝试创建无效的外键 (id, id)。复合索引中涉及的列(一个涉及多于一列)不能是同一列,重复。至少我从未见过。

创建两个单独的外键,一个用于 FROM,另一个用于 TO,每个都指向用户。

于 2013-03-11T12:33:47.483 回答