MySQL数据库#4

外键(表与表之间的关系)

什么是外键:

        外键是指关系数据库中一个表中的字段,它与另一个表的主键相关联,用于建立两个表之间的关联关系。外键可以用来确保数据的完整性,通过检查关联表的主键来限制对于某个表的操作。当对主表进行操作时,外键的限制也会自动作用到从表中。这可以保证数据在不同表之间的一致性,防止出现不合理的数据关系。在使用外键时,需要定义好关系表之间的关联规则,包括指定主表和从表,以及主表和从表之间的关联字段等。

举例:

我们可以建立一个表:

MySQL数据库#4_第1张图片

当我们需要修改部门表的名称的话,则需把工程部里的所有人都得改,牵一发而动全身,很麻烦,

第二个就是这个表的表达不清晰,不清楚是员工表还是部门表,表内的相关字段一直重复存储,

表的扩展性很差,所以我们要想解决,就需要分成两张表

emp 和dep 但是这样的话表与表之后就没有关联了。所以我们需要外键来建立关系。

表与表之间的关系

表与表之间的关系有:

一对多:

想要确认是否为一对多关系需要换位思考法

我们以员工表与部门表为例:

我们先看员工表,一个员工是否可以有多个部门?不能

部门是否可以有多个员工?可以

根据以上分析我们得知,员工表不可以,部门表可以,表之间的关系就是:一对多,在表关系中并没有多对一。我们针对一对多,外键字段就要在多的一方,也就是部门表

如何在mysql表中建立一对多的关系:先把基础的字段建立出来,然后在考虑外键字段

create table emp(
    id int primary key auto_increment,
    name varchar(32),
    age int,
    dep_id int,
    foreign key(dep_id) references dep(id),on update cascade , on delete cascade # 让两张表建立了外键关系
   on update cascaed # 级联更新
    on delete cascade # 级联删除
);

MySQL数据库#4_第2张图片

create table dep(
    id int primary key auto_increment,
    dep_name varchar(32),
    dep_desc varchar(32)
);

多对多

        以图书表和作者表为例
我们站在图书表的角度


    问:一本图书能不能有多个作者?
    答:可以
我们再站在作者表的角度


    问:一个作者能不能写多本书
    答:可以
得出结论:如果两个都可以,那么表关系就是'多对多'


针对于多对多的表关系,外键字段建在第三张表中

create table book(
    id int primary key auto_increment,
    title varchar(32),
    price decimal(8,2)
);

create table author(
    id int primary key auto_increment,
    name varchar(32),
    addr varchar(32)
);

create table book2author(
    id int primary key auto_increment,
    book_id int,
    author_id int,
    foreign key(book_id) references author(id) # 让两张表建立了外键关系
    on update cascade # 级联更新
    on delete cascade, # 级联删除
    foreign key(author_id) references book(id) # 让两张表建立了外键关系
    on update cascade # 级联更新
    on delete cascade
);

输入完之后查看

MySQL数据库#4_第3张图片

insert into book(title, price) values('朝花夕拾', 50);
insert into book(title, price) values('骆驼祥子', 2000);

MySQL数据库#4_第4张图片
insert into author(name, addr) values('luxun', 'beijing');
insert into author(name, addr) values('shuqingchun', 'liaoningi');

MySQL数据库#4_第5张图片

insert into book2author(book_id, author_id) values(1, 1);
insert into book2author(book_id, author_id) values(1, 2);
insert into book2author(book_id, author_id) values(2, 1);
insert into book2author(book_id, author_id) values(2, 2);

MySQL数据库#4_第6张图片

注意事项:

      1.   在创建表的时候 需要先创建被关联表(没有外键字段的表)

      2.在插入新数据的时候 应该先确保被关联表中有数据


3.在插入新数据的时候 外键字段只能填写被关联表中已经存在的数据


4.在修改和删除被关联表中的数据的时候 无法直接操作


    如果想要数据之间自动修改和删除需要添加额外的配置

一对一的关系

create table author1(
    id int primary key auto_increment,
    name varchar(32),
    gender varchar(32),
    author_detail_id int unique,
    foreign key(author_detail_id) references author_detail(id)
    on update cascade
    on delete cascade
);


create table author_detail(
    id int primary key auto_increment,
    qq varchar(32),
    email varchar(32)
);
 

        

多表查询

        我们之前所查的都是单表查询,

多表查询的思路是

  子查询

        查询keivn的部门名称的话我们需要先查询kevin所在的部门id

select dep_id from emp where name='kevin'

       之后再用查出的id去和dep表中查询部门名称

select *from dep where id=(select dep_id from emp where name='kevin');

所以子查询实际就是一条sql的执行结果是另一条sql语句的执行条件。

MySQL数据库#4_第7张图片

多表查询

        多表查询:把多张有关系的表连接成一张大的虚拟表,连接出来的虚拟表不是实际存在的,它是在内存中存储,是按照单表查询。

我们先准备下数据;

create table dep(
    id int primary key auto_increment,
    name varchar(20) 
);

create table emp(
    id int primary key auto_increment,
    name varchar(20),
    sex enum('male','female') not null default 'male',
    age int,
    dep_id int
);


如果两张表没有建立强制的约束关系,就使用逻辑意义上的关联
#插入数据
insert into dep values
(200,'技术'),
(201,'人力资源'),
(202,'销售'),
(203,'运营'),
(205,'保洁')
;

insert into emp(name,sex,age,dep_id) values
('jason','male',18,200),
('egon','female',48,201),
('kevin','male',18,201),
('nick','male',28,202),
('owen','male',18,203),
('jerry','female',18,204);

创建完毕

MySQL数据库#4_第8张图片

        

多表查询

        select *from emp,dep where dep_id=dep.id;

专业的连表语法;

inner join:内连接,查询两张表都有的数据

left join :左连接,以左表为基准,查询左表内所有的数据,右表没有的数据使用NULL代替

right : 右连接,以右表为基准,查询右表内所有的数据,左表没有的数据使用NULL代替

union :连接两个sql语句的结果

select * from emp left join dep on emo.dep_id=dep.id

union

select * from dep right jojn dep on dep.emo_id=dep,id

多表查询练习题

1、查询所有的课程的名称以及对应的任课老师姓名


2、查询平均成绩大于八十分的同学的姓名和平均成绩


3、查询没有报李平老师课的学生姓名

        
4、查询挂科超过两门(包括两门)的学生姓名和班级

你可能感兴趣的:(数据库,mysql,oracle)