数据库学习笔记7-外键 | 8月更文挑战

560 阅读3分钟

这是我参与8月更文挑战的第7天,活动详情查看:8月更文挑战

外键

  • 外键:foreign key,外面的键,键不在自己表中
  • 如果一张表中有一个字段(非主键)指向另外一张表的主键,那么将该字段称之为外键

增加外键

  • 外键可以在创建表的时候或者创建表之后增加,但是要考虑数据的问题

  • 一张表可以有多个外键

  • 创建表的时候增加外键,在所有的表字段之后增加外键, foreign key(外键字段) references 外部表(主键字段)

create table my_foreign1(
	id int primary key auto_increment,
	name varchar(20) not null comment '学生姓名',
	c_id int comment '班级id',	-- 普通字段
	-- 增加外键
	foreign key(c_id) references my_class(id)
)charset utf8;
  • 在新增表之后增加外键:修改表结构
  • alter table 表名 add [constraint 外键名字] foreign key(外键字段) references 父表(主键字段)
-- 创建表
create table my_foreign2(
id int primary key auto_increment,
name varchar(20) not null comment '学生姓名',
c_id int comment '班级id'	-- 普通字段
)charset utf8;

-- 增加外键
alter table my_foreign2 add
-- 指定外键名
constraint student_class_1
-- 指定外键字段
foreign key(c_id)
-- 引用父表主键
references my_class(id);

更新、删除外键

  • 外键不能修改,只能先删除后新增
  • alter table 表名 drop foreign key 外键名
  • 一张表中可以有多个外键,但是名字不能重复,删除的时候必须指定外键的名字
-- 删除外键
alter table my_foreign1 drop  foreign key my_foreign1_ibfk_1;

外键作用

  • 外键默认的作用有两种,一个是对父表,一个对字表(外键字段所在的表)
  • 对子表约束,字表数据进行操作(增和改)的时候,如果对应的外键字段在父表找不到对应的匹配,则操作会失败(约束字表数据操作)

-- 插入数据:外键字段在父表中不存在
insert into my_foreign2 values(null,'张自忠',4); -- 没有4班级
insert into my_foreign2 values(null,'项羽',1); 
insert into my_foreign2 values(null,'刘邦',2); 
insert into my_foreign2 values(null,'韩信',2); 
  • 对父表约束:父表数据进行操作(删、改都必须涉及到主键本身),如果对应的主键在子表中已经被数据引用,那么就不允许操作
-- 更新父表记录
update my_class set id = 4 where id = 1; -- 失败: id=1记录已经被学生引用
update my_class set id = 4 where id = 3; -- 可以: 没有引用

外键条件

  1. 外键要存在:首先必须保证表的存储引擎是innodb(默认的存储引擎,如果不是innodb存储引擎,那么外键可以创建成功,但是没有约束效果
  2. 外键字段的字段类型(列类型)必须与父表的主键类型完全一致
  3. 一张表中的外键名字不能重复
  4. 增加外键的字段(数据已经存在),必须保证数据与父表主键要求对应

外键约束

  • 外键约束就是外键的作用
  • 外键约束有三种约束模式,都是针对父表的约束
    • District: 严格模式(默认的), 父表不能删除或者更新一个已经被子表数据引用的记录
    • Cascade: 级联模式: 父表的操作, 对应子表关联的数据也跟着被删除
    • Set null: 置空模式: 父表的操作之后,子表对应的数据(外键字段)被置空
  • 通常的一个合理的做法(约束模式): 删除的时候子表置空, 更新的时候子表级联操作指定模式的语法
  • Foreign key(外键字段) references 父表(主键字段) on delete set null on update cascade
  • 更新操作: 级联更新
  • 删除操作: 置空