24

我想知道这个测试问题。我自己准备了示例并对其进行了测试,但我仍然不确定答案。

具有以下内容:

CREATE TABLE foo (
  id INT PRIMARY KEY AUTO_INCREMENT, 
  name INT
)

CREATE TABLE foo2 (
  id INT PRIMARY KEY AUTO_INCREMENT, 
  foo_id INT REFERENCES foo(id) ON DELETE CASCADE
)

据我所知,答案是:

一种。创建了两个表

虽然也有:

湾。如果删除表 foo2 中 foo_id 为 2 的行,则表 foo 中 id=2 的行被自动删除

d.如果删除表foo中id=2的行,则删除表foo2中所有foo_id=2的行

在我的示例中,我将使用删除语法:

DELETE FROM foo WHERE id = 2;
DELETE FROM foo2 WHERE foo_id = 2;

出于某种原因,我无法找到表之间的任何关系,尽管看起来应该有一个。也许有一些 MySQL 设置或者ON DELETE CASCADE在表创建查询中没有正确使用?我很想知道...

4

2 回答 2

38

答案 d。是正确的,当且仅当存储引擎实际上支持并强制执行外键约束。

如果表是用 来创建的Engine=MyISAM,那么两者都不是。或 d。是正确的。

如果表是用 来创建的Engine=InnoDB,那么d. 是正确的。

笔记:

这对于 InnoDB 是正确的,当且仅当FOREIGN_KEY_CHECKS = 1; 如果FOREIGN_KEY_CHECKS = 0,则DELETE父表 (foo) 中的a不会从子表 (foo2) 中删除引用从父表中删除的行的行。

使用 (1=ON, 0=OFF)的输出验证这一点SHOW VARIABLES LIKE 'foreign_key_checks' (正常默认设置为 ON。)

输出SHOW CREATE TABLE foo将显示表使用的引擎。

的输出SHOW VARIABLES LIKE 'storage_engine'将显示创建表且未指定引擎时使用的默认引擎。

于 2012-08-29T21:27:15.817 回答
14

您确实在两个表之间有关系,它在 foo2 创建命令中: ... foo_id int references foo(id) on delete cascade.

根据MySQL 外键约束参考

CASCADE:删除或更新父表中的行,并自动删除或更新子表中匹配的行。支持 ON DELETE CASCADE 和 ON UPDATE CASCADE。

此外,根据MySQL 外键参考

对于 InnoDB 以外的存储引擎,可以在定义列时使用 REFERENCES tbl_name(col_name) 子句,该子句没有实际效果,仅作为您当前定义的列的备注或注释引用另一个表中的列

因此,由于外键是从子表到父表的,因此它会foo生成一个父表和一个子表,因此如果您使用 InnoDB 或其他一些支持它的存储引擎, foo2删除一行foo会将删除级联到。foo2

于 2012-08-29T20:35:56.713 回答