我正在使用phpMyAdmin设置数据库。我有两个表(foo和bar),在它们的主键上建立索引。我试图在它们之间创建一个关系表(foo_bar),使用它们的主键作为外键。

我用MyISAM创建了这些表,但后来把三个表都换成了InnoDB,因为我读到MyISAM不支持外键。所有id字段都是INT(11)。

当我选择foo_bar表时,单击“关系视图”链接,并尝试将FK列设置为database.foo.id和database.bar。id,它会在每一列旁边说“未定义索引!”

我错过了什么?

澄清/更新

为了简单起见,我想继续使用phpMyAdmin。我目前使用XAMPP,这很容易让我专注于PHP/CSS/Javascript,它附带phpMyAdmin。

此外,虽然我还不能设置显式外键,我有一个关系表,可以执行这样的连接:

SELECT * 
FROM foo 
INNER JOIN foo_bar 
ON foo.id = foo_bar.foo_id 
INNER JOIN bar
ON foo_bar.bar_id = bar.id;

在数据库中没有显式定义fk只是让我感到不舒服。


当前回答

首先设置“存储引擎”为“InnoDB”

然后在结构菜单中启用关系视图选项

其他回答

如果你想使用phpMyAdmin来建立关系,你必须做两件事。首先,您必须在引用表中的外键列上定义一个索引(因此,foo_bar。Foo_id,在你的情况下)。然后,转到关系视图(在引用表中)并选择引用的列(在本例中为foo.id)以及on update和on delete操作。

我认为外键是有用的,如果你有多个表彼此链接,特别是,你的删除脚本将变得非常短,如果你设置正确的引用选项。

编辑:确保两个表都选择了InnoDB引擎。

这是一个老帖子,但回答,因为如果对任何人都有用。

步骤1。您的Db存储引擎设置为InnoDB

步骤2。创建主表

这里customer是主表,customer_id是主键

步骤3。创建外键表并给出索引

这里我们有customer_addresses作为相关表和存储客户地址,所以这里customer_id与客户表的关系

创建表时可以直接选择索引

如果你在创建表时忘记了索引,那么你可以从表的结构选项卡中给出索引,如下所示。

步骤4。一旦索引给出字段,去结构选项卡,并点击关系视图,如下图所示

第5步。现在选择ON DELETE和ON UPDATE你想做的,从当前表中选择列,选择DB (SAME DB),从该表中选择关系表和主键,如下图所示,并保存它

现在检查关系是否成功给出,转到外表数据列表并单击外键值,您将重定向到主表记录,然后关系创建成功。

InnoDB允许你使用ALTER table向表中添加一个新的外键约束:

ALTER TABLE tbl_name
    ADD [CONSTRAINT [symbol]] FOREIGN KEY
    [index_name] (index_col_name, ...)
    REFERENCES tbl_name (index_col_name,...)
    [ON DELETE reference_option]
    [ON UPDATE reference_option]

另一方面,如果MyISAM在你的环境中比InnoDB有优势,你为什么要创建外键约束呢?您可以在应用程序的模型级别上处理这个问题。只需确保要用作外键的列已被索引!

不要忘记这两列应该具有相同的数据类型。

例如,如果一列是INT类型,另一列是tinyint类型,你会得到以下错误:

在[PID列]上创建外键错误(检查数据类型)

这是一篇维基百科文章的摘要。它指定了可以在PHPmyadmin中规定的不同类型的关系。我把它放在这里,因为它与@Nathan关于设置“更新/删除”外键选项的评论有关,但对于评论来说太大了。

级联

每当主(引用)表中的行被删除时(resp。更新后),相应的子(引用)表的行与匹配的外键列将被删除(resp。更新)以及。这被称为级联删除(resp。更新[2])。

限制

如果外键表中存在引用引用表中的值的行,则不能更新或删除该值。类似地,只要外键表中有对该行的引用,就不能删除该行。

不采取行动

NO ACTION和RESTRICT非常相似。NO ACTION和RESTRICT之间的主要区别是,使用NO ACTION时,参考完整性检查是在尝试修改表之后进行的。RESTRICT在尝试执行UPDATE或DELETE语句之前进行检查。如果引用完整性检查失败,这两个引用操作的作用相同:UPDATE或DELETE语句将导致错误。

设置为空

更新或删除引用行的时候,引用行的外键值被设置为NULL。只有在引用表中各自的列为空时才可能这样做。由于NULL的语义,外键列中包含NULL的引用行不需要引用行。

设置默认

与SET NULL类似,当被引用的行被更新或删除时,引用行的外键值被设置为默认列。