我遇到了一个奇怪的问题。我试图添加一个外键到一个引用另一个表,但由于某种原因失败了。以我对MySQL有限的知识,唯一可能怀疑的是,在另一个表上有一个外键引用了我试图引用的表。

我已经在两个表上做了一个SHOW CREATE TABLE查询,sourcecodes_tags是带外键的表,sourcecodes是引用的表。

CREATE TABLE `sourcecodes` (
 `id` int(11) unsigned NOT NULL AUTO_INCREMENT,
 `user_id` int(11) unsigned NOT NULL,
 `language_id` int(11) unsigned NOT NULL,
 `category_id` int(11) unsigned NOT NULL,
 `title` varchar(40) CHARACTER SET utf8 NOT NULL,
 `description` text CHARACTER SET utf8 NOT NULL,
 `views` int(11) unsigned NOT NULL,
 `downloads` int(11) unsigned NOT NULL,
 `time_posted` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
 PRIMARY KEY (`id`),
 KEY `user_id` (`user_id`),
 KEY `language_id` (`language_id`),
 KEY `category_id` (`category_id`),
 CONSTRAINT `sourcecodes_ibfk_3` FOREIGN KEY (`language_id`) REFERENCES `languages` (`id`) ON DELETE CASCADE ON UPDATE CASCADE,
 CONSTRAINT `sourcecodes_ibfk_1` FOREIGN KEY (`user_id`) REFERENCES `users` (`id`) ON DELETE CASCADE ON UPDATE CASCADE,
 CONSTRAINT `sourcecodes_ibfk_2` FOREIGN KEY (`category_id`) REFERENCES `categories` (`id`) ON DELETE CASCADE ON UPDATE CASCADE
) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=latin1

CREATE TABLE `sourcecodes_tags` (
 `sourcecode_id` int(11) unsigned NOT NULL,
 `tag_id` int(11) unsigned NOT NULL,
 KEY `sourcecode_id` (`sourcecode_id`),
 KEY `tag_id` (`tag_id`),
 CONSTRAINT `sourcecodes_tags_ibfk_1` FOREIGN KEY (`tag_id`) REFERENCES `tags` (`id`) ON DELETE CASCADE ON UPDATE CASCADE
) ENGINE=InnoDB DEFAULT CHARSET=latin1

下面是生成错误的代码:

ALTER TABLE sourcecodes_tags ADD FOREIGN KEY (sourcecode_id) REFERENCES sourcecodes (id) ON DELETE CASCADE ON UPDATE CASCADE

当前回答

同样的问题我遇到过三次。在每种情况下,都是因为我的一个(或多个)记录不符合新的外键。在尝试添加外键本身之前,您可能希望更新现有记录以遵循外键的语法约束。下面的例子通常可以隔离问题记录:

SELECT * FROM (tablename)
    WHERE (candidate key) <> (proposed foreign key value) 
        AND (candidate key) <> (next proposed foreign key value)

对于外键中的每个值,在查询中重复AND(候选键)<>(下一个建议的外键值)。

如果您有大量的记录,这可能会很困难,但如果您的表相当小,这应该不会花费太长时间。我在SQL语法方面不是很出色,但这始终为我隔离了问题。

其他回答

最后,我删除了表中的所有数据,并再次运行alter。它的工作原理。不是很聪明,但它节省了很多时间,尤其是你的应用程序还处于开发阶段,没有任何客户数据。

我今天也遇到了同样的问题。我测试了四个方面,其中一些已经在这里提到了:

你的子列中是否存在父列中不存在的值(除了NULL,如果子列为空) 子列和父列是否具有相同的数据类型? 您所引用的父列上是否有索引?MySQL似乎出于性能原因需要这样做(http://dev.mysql.com/doc/refman/5.5/en/create-table-foreign-keys.html) 这个为我解决了这个问题:两个表是否具有相同的排序规则?

一个表是UTF-8格式,另一个表是iso-something格式。这并没有起作用。将iso-table更改为UTF-8排序规则后,可以毫无问题地添加约束。在我的例子中,phpMyAdmin甚至没有在创建外键约束的下拉菜单中显示iso-encoding的子表。

这也发生在设置父外键时。Id到child。列如果是子元素。列的值已经为0,并且没有父元素。Id值为0

你需要确保每个孩子。列为NULL或在parent.id中存在值

现在我读了nos写的声明,这就是他所确认的。

对我来说,这个问题有点不同,而且超级容易检查和解决。

你必须确保你的两个表都是InnoDB。如果其中一个表(即引用表)是MyISAM,约束将失败。

    SHOW TABLE STATUS WHERE Name =  't1';

    ALTER TABLE t1 ENGINE=InnoDB;

我正在准备这个解决方案,这个例子可能会有帮助。

我的数据库有两个表(email和credit_card),它们的id都有主键。另一个表(客户端)将这些表id引用为外键。除了客户数据之外,我有理由拥有电子邮件。

首先,我为引用的表(email, credit_card)插入行数据,然后获得每个表的ID,这些ID在第三个表(客户端)中需要。

如果你不先在引用的表中插入行,当你在第三个引用外键的表中插入新行时,MySQL将不能进行相应的操作。

如果首先插入引用表的引用行,然后插入引用外键的行,则不会发生错误。

希望这能有所帮助。