我正在使用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只是让我感到不舒服。


当前回答

您也可以使用SQL命令来完成,如下所示。

ALTER TABLE employees
    ADD CONSTRAINT fk_companyid FOREIGN KEY (companyid)
    REFERENCES companies (id)
    ON DELETE CASCADE;

在本例中,如果删除了companies中的一行,那么具有该companyid的所有员工也将被删除。

其他回答

从MySQL官方文档https://dev.mysql.com/doc/refman/8.0/en/create-table-foreign-keys.html:

MySQL要求在外键和引用键上建立索引,以便 外键检查可以快速且不需要表扫描。

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有优势,你为什么要创建外键约束呢?您可以在应用程序的模型级别上处理这个问题。只需确保要用作外键的列已被索引!

您也可以使用SQL命令来完成,如下所示。

ALTER TABLE employees
    ADD CONSTRAINT fk_companyid FOREIGN KEY (companyid)
    REFERENCES companies (id)
    ON DELETE CASCADE;

在本例中,如果删除了companies中的一行,那么具有该companyid的所有员工也将被删除。

phpMyAdmin允许您使用“关系”视图定义外键。但是由于MySQL只支持对“INNO DB”表的外部约束,所以第一步是确保你所使用的表是这种类型的。

要设置一个外键,使CHILD表中的PID列引用PARENT表中的ID列,可以执行以下操作:

For both tables, go to the operations tab and change their type to "INNO DB" Make sure ID is the primary key (or at least an indexed column) of the PARENT table. In the CHILD table, define an index for the PID column. While viewing the structure tab of the CHILD table, click the "relation view" link just above the "add fields" section. You will be given a table where each row corresponds to an indexed column in your CLIENT table. The first dropdown in each row lets you choose which TABLE->COLUMN the indexed column references. In the row for PID, choose PARENT->ID from the dropdown and click GO.

通过对CHILD表进行导出,您应该看到已经为PID列创建了一个外键约束。

对于那些刚接触数据库....的人并且需要ALTER一个现有的表。很多事情看起来都很简单,但总有一些……在A和B之间。

在做其他事情之前,先看看这个。

Make sure you have P_ID (parent ID on both parent and child table). Of course it will be already filled in the parent. Not necessarily in the child in a true and final way. So for instance P_ID #3 (maybe many times in the child table will be pointing to original P_ID at parent table). Go to SQL tab (I am using phpMyAdmin, should be similar in other ones) and do this command: ALTER TABLE child_table_name ADD FOREIGN KEY (P_ID) REFERENCES parent_table_name (P_ID) Click on child table, than structure, finally on relational view. Finish your DB planning there. There was a nice answer before this one about cascade, restrict, etc. Of course it could be done by commands...