我遇到了一个奇怪的问题。我试图添加一个外键到一个引用另一个表,但由于某种原因失败了。以我对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

当前回答

清空两个表的数据并运行该命令。它会起作用的。

其他回答

很可能您的sourcecodes_tags表包含的sourcecode_id值在您的sourcecodes表中不再存在。你得先把它们处理掉。

下面是一个可以找到这些id的查询:

SELECT DISTINCT sourcecode_id FROM 
   sourcecodes_tags tags LEFT JOIN sourcecodes sc ON tags.sourcecode_id=sc.id 
WHERE sc.id IS NULL;

我的MySQL数据库也有同样的问题,但最后,我找到了一个适合我的解决方案。 因为在我的表中,从mysql的角度来看一切都很好(两个表都应该使用InnoDB引擎,每列的数据类型应该是相同的类型,参与外键约束)。 我所做的唯一一件事是禁用外键检查,然后在执行外键操作后启用它。 我采取的步骤:

SET foreign_key_checks = 0;
alter table tblUsedDestination add constraint f_operatorId foreign key(iOperatorId) references tblOperators (iOperatorId); Query
OK, 8 rows affected (0.23 sec) Records: 8  Duplicates: 0  Warnings: 0
SET foreign_key_checks = 1;

你只需要回答一个问题

您的表已经存储数据了吗?(特别是表中包含外键。)

如果答案是肯定的,那么您需要做的唯一一件事就是删除所有记录,然后您可以自由地向表中添加任何外键。

删除指令:从子表(包括外键表)到父表。

不能在数据条目后添加外键的原因是由于表不一致,您将如何处理一个新的外键对前数据填充的表?

如果答案是否定的,那么按照其他指示进行。

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

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

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

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

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

希望这能有所帮助。

你可以试试这个例子

 START TRANSACTION;
 SET foreign_key_checks = 0;
 ALTER TABLE `job_definers` ADD CONSTRAINT `job_cities_foreign` FOREIGN KEY 
 (`job_cities`) REFERENCES `drop_down_lists`(`id`) ON DELETE CASCADE ON UPDATE CASCADE;
 SET foreign_key_checks = 1;
 COMMIT;

注意:如果您正在使用phpmyadmin,只需取消启用外键检查

为例

希望这个解决方案能解决你的问题:)