MySQL DDL操作--------外键最佳实战

1. 背景

* MySQL有两种常用的引擎类型MyISAM和InnoDB。目前只有InnoDB引擎类型支持外键约束。

* 本表的列必须与外键类型相同, 外键必须是外表的主键

* 设置外建的列不能设置 NO NULL 字段属性。

2. 外建作用

* 使两张表形成关联,外键只能引用外表中的列的值

* 保持数据一致性,完整性,控制存储在外键表中的数据

3. 外键实验 [ 员工 --> 部门 ]

* 创建外键依赖的外表 departments

mysql> CREATE TABLE departments(
    -> id BIGINT PRIMARY KEY NOT NULL AUTO_INCREMENT,
    -> name VARCHAR(64) NOT NULL
    -> )ENGINE=INNODB CHARSET=utf8mb4;
Query OK, 0 rows affected (0.05 sec)

* 创建员工表 empoyees, 并建立外键

指定外键关键字: FOREIGN KEY(列名)

引用外键关键字: REFERENCES <外键表名>(外键列名) 

事件触发限制:  [默认] no action

ON DELETE  SET NULL   当外表字段删除时,本表设置为NULL(空值) [ 不推荐设置此荐 ]

ON UPDATE CASCADE   当外表字段更新时,本表外键级联更新

ON DELETE/UPDATE SET DEFAULT 当有事件触发时,设置为默认值

    ON DELETE/UPDATE RESTRICT      当有事件触发时,限制外表中的外键改动

mysql> CREATE TABLE empoyees( 
    -> id BIGINT PRIMARY KEY NOT NULL AUTO_INCREMENT,
    -> sex ENUM(‘M‘, ‘F‘) NOT NULL,
    -> age INT NOT NULL,
    -> department BIGINT,
    -> FOREIGN KEY(department) 
    -> REFERENCES departments(id) 
    -> ON DELETE SET NULL
    -> ON UPDATE CASCADE
    -> )ENGINE=INNODB CHARSET=utf8mb4;
Query OK, 0 rows affected (0.12 sec)

* 对部门表 departments 插入数据

mysql> INSERT INTO departments SELECT NULL, ‘dev‘;
Query OK, 1 row affected (0.01 sec)
Records: 1  Duplicates: 0  Warnings: 0

mysql> INSERT INTO departments SELECT NULL, ‘test‘;
Query OK, 1 row affected (0.01 sec)
Records: 1  Duplicates: 0  Warnings: 0

mysql> INSERT INTO departments SELECT NULL, ‘ops‘;
Query OK, 1 row affected (0.01 sec)
Records: 1  Duplicates: 0  Warnings: 0

* 查看部门表数据

mysql> SELECT * FROM departments;
+----+------+
| id | name |
+----+------+
|  1 | dev  |
|  2 | test |
|  3 | ops  |
+----+------+
3 rows in set (0.00 sec)

* 插入正常数据 (部门编号存在于部门表中)

mysql> INSERT INTO empoyees SELECT NULL, ‘M‘, 22, 2;
Query OK, 1 row affected (0.01 sec)
Records: 1  Duplicates: 0  Warnings: 0

mysql> SELECT * FROM empyees;
ERROR 1146 (42S02): Table ‘mytest.empyees‘ doesn‘t exist
mysql> SELECT * FROM empoyees;
+----+-----+-----+------------+
| id | sex | age | department |
+----+-----+-----+------------+
|  1 | M   |  22 |          2 |
+----+-----+-----+------------+
1 row in set (0.01 sec)

* 插入非正常数据 (部门编号不存在于部门表中) [ 部门表中不存在id为4的列 ]

mysql> INSERT INTO empoyees SELECT NULL, ‘M‘, 22, 4;
ERROR 1452 (23000): Cannot add or update a child row: a foreign key constraint fails (`mytest`.`empoyees`, CONSTRAINT `empoyees_ibfk_1` FOREIGN KEY (`department`) REFERENCES `departments` (`id`) ON DELETE SET NULL ON UPDATE CASCADE)

* 修改部门表数据 [ 员工表中已有数据关联部门表中id为2 ]

mysql> SELECT * FROM empoyees;
+----+-----+-----+------------+
| id | sex | age | department |
+----+-----+-----+------------+
|  1 | M   |  22 |          2 |
+----+-----+-----+------------+
1 row in set (0.01 sec)

mysql> UPDATE departments SET id = 4 WHERE id=2;
Query OK, 1 row affected (0.04 sec)
Rows matched: 1  Changed: 1  Warnings: 0

mysql> SELECT * FROM empoyees;
+----+-----+-----+------------+
| id | sex | age | department |
+----+-----+-----+------------+
|  1 | M   |  22 |          4 |
+----+-----+-----+------------+
1 row in set (0.01 sec)

* 删除部门表数据 [ 员工表中已有数据关联部门表中id为4 ]

mysql> SELECT * FROM empoyees;
+----+-----+-----+------------+
| id | sex | age | department |
+----+-----+-----+------------+
|  1 | M   |  22 |          2 |
+----+-----+-----+------------+
1 row in set (0.01 sec)

mysql> DELETE FROM departments WHERE id = 4;
Query OK, 1 row affected (0.01 sec)

mysql> SELECT * FROM empoyees;
+----+-----+-----+------------+
| id | sex | age | department |
+----+-----+-----+------------+
|  1 | M   |  22 |       NULL |
+----+-----+-----+------------+
1 row in set (0.00 sec)

4. 总结

以需求驱动技术,技术本身没有优略之分,只有业务之分。

时间: 2024-10-25 20:04:59

MySQL DDL操作--------外键最佳实战的相关文章

[原创]MYSQL中利用外键实现级联删除和更新

MySQL中利用外键实现级联删除.更新 MySQL支持外键的存储引擎只有InnoDB,在创建外键的时候,要求父表必须有对应的索引,子表在创建外键的时候也会自动创建对应的索引.在创建索引的时候,可以指定在删除.更新父表时,对子表进行的相应操作,包括RESTRICT.NO ACTION.SET NULL和CASCADE.其中RESTRICT和NO ACTION相同,是指在子表有关联记录的情况下父表不能更新:CASCADE表示父表在更新或者删除时,更新或者删除子表对应记录:SET NULL则是表示父表

mysql foreign key(外键) 说明与实例

一,什么是foreign key,及其完整性 个人觉得,foreign key就是表与表之间的某种约定的关系,由于这种关系的存在,我们能够让表与表之间的数据,更加的完整,关连性更强.关于完整性,关连性我举个例子,大家就会明白了. 有二张表,一张是用户表,一张是订单表: 1,如果我删除了用户表里的用户,那么订单表里面根这个用户有关的数据,就成了无头数据了,不完整了. 2,如果我在订单表里面,随便插入了一条数据,这个订单在用户表里面,没有与之对应的用户.这样数据也不完整了. 如果有外键的话,就方便多

MySQL学习9:详解MySQL中的外键约束

本篇主要讲解MySQL中的外键约束. 一约束概述 创建约束的目的就是保证数据的完整性和一致性.约束根据约束针对的字段的数目的多少划分为表级约束和列级 约束. 如果约束按照功能来划分的话刻印划分为:NOT NULL(非空约束).PRIMARY KEY(主键约束).UNIQUE KEY(唯一约束).DEFAULT(默认约束)和FOREIGN KEY(外键约束). 在前面我们已经初步涉及到了除外键约束之外的四个约束,这一次我们来谈谈最复杂的外键约束. 二外键约束的要求 FOREIGN KEY(外键约束

mysql怎么定义外键

数据库mysql 建立外键的前提: 本表的列必须与外键类型相同(外键必须是外表主键). 外键作用: 使两张表形成关联,外键只能引用外表中的列的值! 指定主键关键字: foreign key(列名) 引用外键关键字: references <外键表名>(外键列名) 事件触发限制: on delete和on update , 可设参数cascade(跟随外键改动), restrict(限制外表中的外键改动),set Null(设空值),set Default(设默认值),[默认]no action

Mysql无法创建外键的原因

MySQL无法创建外键的原因 1.  两个字段的类型或者大小不严格匹配.例如,如果一个是int(10),那么外键也必须设置成int(10),而不是int(11),也不能是tinyint.另外,你还必须确定两个字段是否一个为 signed,而另一个又是unsigned(即:无符号),这两字段必须严格地一致匹配,更多关于signed和unsigned的信息,请参阅:http://www.verysimple.com/blog/?p=57 2. 试图设置外键的字段没有建立起索引,或者不是一个prima

MySQL里创建外键时错误的解决

--MySQL里创建外键时错误的解决--------------------------------2014/04/30在MySQL里创建外键时(Alter table xxx add constraint fk_xxx foreign key),提示错误,但只提示很简单的信息:ERROR 1005 (HY000): Can't create table '.\env_mon\#sql-698_6.frm' (errno: 150).根本起不到解决问题的作用.要看错误的详细提示,可以使用命令:(

转!!!Mysql无法创建外键的原因

在Mysql中创建外键时,经常会遇到问题而失败,这是因为Mysql中还有很多细节需要我们去留意,我自己总结并查阅资料后列出了以下几种常见原因. 1.  两个字段的类型或者大小不严格匹配.例如,如果一个是int(10),那么外键也必须设置成int(10),而不是int(11),也不能是tinyint.另外,你还必须确定两个字段是否一个为 signed,而另一个又是unsigned(即:无符号),这两字段必须严格地一致匹配,更多关于signed和unsigned的信息,请参阅:http://www.

MySQL数据库建立外键失败的原因总结

http://database.51cto.com/art/201108/280088.htm 在MySQL数据库创建外键时,经常会发生一些错误,这是一件很令人头疼的事.一个典型的错误就是:Can’t create table... 的错误.在很多实例中,这种错误的发生都是因为mysql一直以来都不能很好的支持的关系的问题, 更不幸的是它也并没有指明到底是哪一个问题会导致上面那种错误,下面我把导致这个可怕的150错误的常见原因列出来了,并且我以可能性的大小作了排序,已知的原因: 1.两个字段的类

(转)mysql 无法设置外键的原因总结

在Mysql中创建外键时,经常会遇到问题而失败,这是因为Mysql中还有很多细节需要我们去留意,我自己总结并查阅资料后列出了以下几种常见原因. 1.  两个字段的类型或者大小不严格匹配.例如,如果一个是int(10),那么外键也必须设置成int(10),而不是int(11),也不能是tinyint.另外,你还必须确定两个字段是否一个为 signed,而另一个又是unsigned(即:无符号),这两字段必须严格地一致匹配,更多关于signed和unsigned的信息,请参阅:http://www.