MySQL学习笔记(15)-外键

外键

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

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

增加外键

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

一张表可以有多个外键

desc my_class;
select * from my_class;
-- 创建外键
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;
desc my_foreign1;
show create table my_foreign1;



在新增表之后增加外键:修改表结构

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;
desc my_foreign2;
desc my_foreign2;
-- 增加外键
alter table my_foreign2 add
-- 指定外键名
constraint student_class_1
-- 指定外键字段
foreign key(c_id)
-- 引用父表主键
references my_class(id);
desc my_foreign2;
show create table my_foreign2;


修改外键&删除外键

外键不可修改:只能先删除后新增。

alter table 表名 drop foreign key 外键名; --一张表中可以有多个外键但是名字不能相同

desc my_foreign1;
alter table my_foreign1 drop foreign key my_foreign1_ibfk_1;
desc my_foreign1;
show create table  my_foreign1; 

select * from my_class;
-- 插入数据:外键字段在父表中不存在
insert into my_foreign2 values(null,'张三',4) -- 没有4班级

insert into my_foreign2 values(2,'项羽',1);
insert into my_foreign2 values(3,'刘邦',2);
insert into my_foreign2 values(4,'韩信',2);
select * from my_class;
-- 更行父表记录
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.增加外键的字段(数据已经存在),必须保证数据与父表主键要求对应。

select * from my_foreign1;
select * from my_class;
-- 增加外键
alter table my_foreign1 add foreign key(c_id) references my_class(id);

外键约束

所谓外键约束:就是指外键的作用。

之前所讲的外键作用:是默认的作用;其实可以通过对外键的需求,进行定制操作。

外键约束有三种约束模式:都是针对父表的约束

    District:严格模式(默认的),父表不能删除或者更新一个已经被子表数据引用的记录

    Cascade:级联模式:父表的操作,对应子表关联的数据也跟着被删除

    Set null:置空模式:父表的操作之后,子表对应的数据(外键字段)被置空

通常的一个合理的做法(约束模式):删除的时候子表置空,更新的时候子表级联操作指定模式的语法

Foreign key(外键字段) references 父表(主键字段)on delete set null on update cascade;

-- 创建外键:指定模式:删除置空,更新级联
create table my_foreign3(
id int primary key auto_increment,
name varchar(20) not null,
c_id int,
-- 增加外键
foreign key(c_id)
-- 引用表
references my_class(id)
-- 指定删除模式
on delete set null
-- 指定更新模式
on update cascade)charset utf8;
-- 插入数据
insert into my_foreign3 values(null,'刘备',1),
(null,'曹操',1),
(null,'孙权',1),
(null,'诸葛亮',2),
(null,'周瑜',2);


更新操作:级联更新

select * from my_foreign3;
select * from my_class;
-- 更新父表主键
update my_class set id = 7 where id = 6;
select * from my_class;
select * from my_foreign3;

删除置空

select * from my_foreign3;
-- 删除父表主键
delete from my_class where id = 7;
select * from my_class;
select * from my_foreign3;

外键虽然很强大,能够进行各种约束:但是对应PHP来讲,外键的约束降低了PHP对数据的可控性:通常在实际开发中,很少使用外键来处理。





猜你喜欢

转载自blog.csdn.net/qq_38826019/article/details/80627842