一、什么是主键、外键:
关系型数据库中的一条记录中有若干个属性,若其中某一个属性组(注意是组)能唯一标识一条记录,该属性组就可以成为一个主键
比如
学生表(学号,姓名,性别,班级)
其中每个学生的学号是唯一的,学号就是一个主键
课程表(课程编号,课程名,学分)
其中课程编号是唯一的,课程编号就是一个主键
成绩表(学号,课程号,成绩)
成绩表中单一一个属性无法唯一标识一条记录,学号和课程号的组合才可以唯一标识一条记录,所以 学号和课程号的属性组是一个主键
成绩表中的学号不是成绩表的主键,但它和学生表中的学号相对应,并且学生表中的学号是学生表的主键,则称成绩表中的学号是学生表的外键
同理 成绩表中的课程号是课程表的外键
定义主键和外键主要是为了维护关系数据库的完整性,总结一下:
1.主键是能确定一条记录的唯一标识,比如,一条记录包括身份正号,姓名,年龄。
身份证号是唯一能确定你这个人的,其他都可能有重复,所以,身份证号是主键。
2.外键用于与另一张表的关联。是能确定另一张表记录的字段,用于保持数据的一致性。
比如,A表中的一个字段,是B表的主键,那他就可以是A表的外键。
二、 主键、外键的区别:
| 主键 | 外键 | 索引 | ||||
定义: | 唯一标识一条记录,不能有重复的,不允许为空 | 表的外键是另一表的主键, 外键可以有重复的, 可以是空值 | 该字段没有重复值,但可以有一个空值 | ||||
作用: | 用来保证数据完整性 | 用来和其他表建立联系用的 | 是提高查询排序的速度 | ||||
个数: | 主键只能有一个 | 一个表可以有多个外键 | 一个表可以有多个惟一索引 |
三,作用
外键取值规则:空值或参照的主键值。
(1)插入非空值时,如果主键表中没有这个值,则不能插入。
(2)更新时,不能改为主键表中没有的值。
(3)删除主键表记录时,可以在建外键时选定外键记录一起级联删除还是拒绝删除。
(4)更新主键记录时,同样有级联更新和拒绝执行的选择。
SQL的主键和外键就是起约束作用。
四,让外键关联值随主键变化而变化
要想让两张表实现级联操作可在外键定义时,使用以下语法
该语法 可以在 CREATE TABLE 和 ALTER TABLE 时使用,如果不指定CONSTRAINT symbol,MYSQL会自动生成一个名字。
ON DELETE、ON UPDATE表示事件触发限制,可设参数:
RESTRICT(限 制外表中的外键改动)
CASCADE(跟随外键改动)c
SET NULL(设空值)
SET DEFAULT(设默认值)
NO ACTION(无动作,默认的)
参考某位大神的实例
CREATE TABLE `dage` (
2 `id` int ( 11 ) NOT NULL auto_increment,
3 `name` varchar ( 32 ) default '' ,
4 PRIMARY KEY (`id`)
5 ) ENGINE = InnoDB DEFAULT CHARSET = latin1;
6
7 CREATE TABLE `xiaodi` (
8 `id` int ( 11 ) NOT NULL auto_increment,
9 `dage_id` int ( 11 ) default NULL ,
10 `name` varchar ( 32 ) default '' ,
11 PRIMARY KEY (`id`),
12 KEY `dage_id` (`dage_id`),
13 CONSTRAINT `xiaodi_ibfk_1` FOREIGN KEY (`dage_id`) REFERENCES `dage` (`id`)
14 ) ENGINE = InnoDB DEFAULT CHARSET = latin1;
插入个大哥:
1 mysql > insert into dage(name) values ( ' 铜锣湾 ' );
2 Query OK, 1 row affected ( 0.01 sec)
3 mysql > select * from dage;
4 + -- --+--------+
5 | id | name |
6 + -- --+--------+
7 | 1 | 铜锣湾 |
8 + -- --+--------+
9 1 row in set ( 0.00 sec)
插入个小弟:
1 mysql > insert into xiaodi(dage_id,name) values ( 1 , ' 铜锣湾_小弟A ' );
2 Query OK, 1 row affected ( 0.02 sec)
3
4 mysql > select * from xiaodi;
5 + -- --+---------+--------------+
6 | id | dage_id | name |
7 + -- --+---------+--------------+
8 | 1 | 1 | 铜锣湾_小弟A |
9 + -- --+---------+--------------+
把大哥删除:
1 mysql > delete from dage where id = 1 ;
2 ERROR 1451 ( 23000 ): Cannot delete or update a parent row: a foreign key constraint fails (`bstar / xiaodi`, CONSTRAINT `xiaodi_ibfk_1` FOREIGN KEY (`dage_id`) REFERENCES `dage` (`id`))
提示:不行呀,有约束的,大哥下面还有小弟,可不能扔下我们不管呀!
插入一个新的小弟:
1 mysql > insert into xiaodi(dage_id,name) values ( 2 , ' 旺角_小弟A ' );
2 ERROR 1452 ( 23000 ): Cannot add or update a child row: a foreign key constraint fails (`bstar / xiaodi`, CONSTRAINT `xiaodi_ibfk_1` FOREIGN KEY (`dage_id`) REFERENCES `dage` (`id`))
3
提示:小子,想造反呀!你还没大哥呢!
把外键约束增加事件触发限制:
1 mysql > show create table xiaodi;
2
3 CONSTRAINT `xiaodi_ibfk_1` FOREIGN KEY (`dage_id`) REFERENCES `dage` (`id`)
4
5 mysql > alter table xiaodi drop foreign key xiaodi_ibfk_1;
6 Query OK, 1 row affected ( 0.04 sec)
7 Records: 1 Duplicates: 0 Warnings:
8.mysql > alter table xiaodi add foreign key (dage_id) references dage(id) on delete cascade on update cascade ;(注意这里!添加了on delete cascade on update cascade后实现删除大哥,小弟也跟着删除了)
9 Query OK, 1 row affected ( 0.04 sec)
10 Records: 1 Duplicates: 0 Warnings: 0
再次试着把大哥删了:
1 mysql > delete from dage where id = 1 ;
2 Query OK, 1 row affected ( 0.01 sec)
3
4 mysql > select * from dage;
5 Empty set ( 0.01 sec)
6
7 mysql > select * from xiaodi;
8 Empty set ( 0.00 sec)
得,这回对应的小弟也没了,没办法,谁让你跟我on delete cascade了呢!