【MySQL | 第四篇】多表查询

目录

一、多表关系

1.一对N

2.N对N

3.一对一

二、什么是多表查询?

1.概述

2.分类------连接查询

①连接查询------内连接

②连接查询------外连接

③连接查询------自连接

3.分类------联合查询

4.分类------子查询

①标量子查询

②列子查询

③行子查询

④表子查询


一、多表关系

多表关系由三类组成,分别为一对N、N对N、一对一。

1.一对N

案例:部门和员工之间的关系。

关系:一个部门对应多个员工,一个员工对应一个部门。

实现:在多的一方建立外键,指向一的一方的主键。

2.N对N

案例:学生与课程的关系。

关系:一个学生可以选择多门课程,一门课程可以由多个学生选择。

实现:建立一个第三方表,第三方表包含两个外键,分别关联两方的主键。

3.一对一

案例:用户与用户详情的关系。

关系:一对一关系,多用于单表拆分,将一张表的基础字段放在一张表中,其他详情字段放到另一张表中。

实现:在任意一方加入外键,关联另外一方的主键, 并设置外键为唯一(UNIQUE)。

二、什么是多表查询?

1.概述

多表查询:顾名思义,指从多张表中查询数据。

笛卡尔积:笛卡尔乘积是指在数学中,两个集合A和集合B的所有组合情况,在多表查询时,需要消除无效的笛卡尔积

2.分类------连接查询

  • 连接查询
    • 内连接:查询A、B交集部分的数据。
    • 外连接:
      • 左外连接:查询左表所有数据,以及两张表交集部分数据。
      • 右外连接:查询右表所有数据,以及两张表交集部分数据。
    • 自连接:当前表与自身的连接查询,自连接必须使用表别名。
  • 子查询

①连接查询------内连接

查询A、B交集部分的数据。

内连接查询语法:

  • 隐式内连接
sql 复制代码
SELECT 字段列表 FROM 表1, 表2 WHERE 条件......;
  • 显式内连接
sql 复制代码
SELECT 字段列表 FROM 表1 [INNER] JOIN 表2 ON 连接条件......;

案例:

复制代码
查询每个员工的姓名,及关联的部门的名称(隐式内连接)
sql 复制代码
select emp.name,dept.name
from emp, dept
where emp.dept_id=dept.id;
复制代码
查询每个员工的姓名,及关联的部门的名称(显式内连接)
sql 复制代码
select e.name, d.name
from emp e inner join dept d
on e.dept_id = d.id;

②连接查询------外连接

外连接查询语法:

  • 左外连接
sql 复制代码
SELECT 字段列表 FROM 表1 LEFT [OUTER] JOIN 表2 ON 条件...;
  • 右外连接
sql 复制代码
SELECT 字段列表 FROM 表1 RIGHT [OUTER] JOIN 表2 ON 条件...;

案例:

复制代码
查询emp的所有数据,和对应的部门名称
sql 复制代码
select e.*, d.name
from emp e left outer join dept d
on e.dept_id = d.id;

运行结果:

③连接查询------自连接

自连接查询语法:

sql 复制代码
SELECT 字段列表 FROM 表A 别名A JOIN 表A 别名B ON 条件...;

自连接查询,可以是内连接查询,也可以是外连接查询。

案例

复制代码
查询员工及其所属领导的名字
sql 复制代码
select a.name, b.name
from emp a join emp b
on a.managerid = b.id;

运行结果:

复制代码
查询员工及其所属领导的名字,如果员工没有领导,也需要查询出来
sql 复制代码
select a.name '员工', b.name '领导'
from emp a left join emp b
on a.managerid = b.id;

运行结果:

3.分类------联合查询

联合查询-union,union all。

对于union查询,就是把多次查询的结果合并起来,形成一个新的查询结果集。

sql 复制代码
SELECT 字段列表 FROM 表A ......
UNION [ALL]
SELECT 字段列表 FROM 表B ......;

案例:

复制代码
将薪资低于5000的员工,和年龄大于50的员工全部查询出来。
sql 复制代码
select * from emp where salary < 5000
union all
select * from emp where age > 50;

如果要去重,则使用union:

sql 复制代码
select * from emp where salary < 5000
union
select * from emp where age > 50;

注:在使用联合查询时,使用union连接的两条语句的返回的列数必须一致。

4.分类------子查询

概念:SQL语句中嵌套SELECT语句,称为嵌套查询,又称子查询

sql 复制代码
SELECT * FROM t1 WHERE column1 = (SELECT column1 FROM t2);

根据子查询结果不同,分为:

  • 标量子查询(子查询结果为单个值)
  • 列子查询(子查询结果为一列)
  • 行子查询(子查询结果为一行)
  • 表子查询(子查询结果为多行多列)

根据子查询位置,分为:WHERE之后、FROM之后、SELECT之后。

①标量子查询

案例:查询销售部的员工信息。

sql 复制代码
select * from emp where dept_id = (select id from dept where name = '销售部');

②列子查询

子查询结果为一列(可以是多行),这种子查询称为列子查询。

常用的操作符:IN,NOT IN, ANY, SOME,ALL

案例:查询销售部和市场部的所有员工信息

sql 复制代码
select * from dept where dept_id in (select id from dept where name = '销售部' or name = '市场部');

③行子查询

子查询返回的结果为一行(可以是多列),这种子查询称为行子查询。

常用的操作符:= <> IN NOT IN

案例:查询与张无忌的薪资与直属领导相同的员工信息

sql 复制代码
select * from emp where (salary, managerid) = ( select salary, managerid from emp where name = '张无忌' );

④表子查询

子查询返回的结果为多行多列,这种子查询称为表子查询。

常用操作符:IN

案例:查询入职日期是"2006-01-01"之后的员工信息,及其部门信息

sql 复制代码
select e.*, d.* from (select * from emp where entrydate > '2006-01-01') e
left join dept d
on e.dept_id = d.id;
相关推荐
sinat_383437363 小时前
Laravel 8 中实现错误日志与调试日志分离的完整配置指南
jvm·数据库·python
sunshine88511 小时前
财务RPA的深水区应用:超越自动化,迈向智能决策支持
数据库
efir OONA11 小时前
MySQL数据库误删恢复_mysql 数据 误删
数据库·mysql·adb
zhangchaoxies12 小时前
如何在 Go 中安全复制接口指针所指向的值
jvm·数据库·python
陈陈CHENCHEN12 小时前
【数据库】MySQL 8.0.40 至 8.0.44 RPM 方式升级指南
数据库·mysql
m0_7349497913 小时前
怎么利用Navicat进行调整备份文件压缩等级_详细配置与操作步骤
jvm·数据库·python
T.i.s13 小时前
番外续2-MIT-BIH Arrhythmia Database
数据库
有味道的男人13 小时前
AI 效率翻倍:对接 1688 拍立淘接口,商品全量信息一键抓取
数据库
m0_7411733314 小时前
如何处理SQL中的NULL值_使用ISNULL或COALESCE函数
jvm·数据库·python