MySQL多表联查——内连、外连

前言

SQL语句以前一直没当回事,总觉得就是select + where组合,前几天面试被拷打了,生产环境下的多表联查很不熟练,现在补上。

多表联查主要借助连接join实现,将两张表的内容进行关联,具体可分为:内连inner join、外连outer join、交叉连接cross join、全连接full join、自连接和自然连接natural join,常用的主要就是内连、外连和交叉连接三种。

有关连接详细介绍可见:
SQL六大连接解析

连接方式示意图如下:

还是简单示例,多几张表规律也不变,这里只创建学生和班级关联的两张表。

故意设置两条冗余数据,没有完全关联,用于体现不同连接方式的查询结果。

here we go

内连查询

内连接又称等值连接,返回两张表共同的部分,用于得到内部共有的数据,如下图所示:

内连查询inner join

关键字inner join,也可写为joinon为关联条件,多表查询时用,逗号分割,默认使用内连查询,即

sql 复制代码
select * from tab_a, tab_b where 条件;
# 等价于
select  *  from  tab_a inner  join  tab_b on 条件;

直接执行

sql 复制代码
SELECT student.`name`,class.class_name FROM student inner join class ;

显示如下:

缺少关联条件on时,内连查询会退化为交叉查询,结果组合为笛卡尔积的形式返回。

补充:name属于 MySQL 潜在关键字 / 常用内置词,为避免解析混淆,建议用`包围

关联条件 on

使用关联条件进行限定的完整sql语句如下:

sql 复制代码
SELECT student.`name`,class.class_name 
FROM student inner join class 
# 匹配关联规则
on student.class_id=class.class_id;
# 过滤条件仍然使用 where 

此时可正常展示交集的三条数据:

外连查询

外连接是将本表与外部表组合到一起,连接外部,具体可分为左外连接、右外连接,全连接。

左外连接

左外连接left join=left outer join,是以左侧表为准,连接外部,即:

示例语句:

sql 复制代码
SELECT student.`name`,class.class_name 
FROM student left join class 
on student.class_id=class.class_id;

输出结果如下:

右外连接

右外连接则完全相反,保留右表全部数据,左侧没有的数据置为空,这里不重复展示。

全连接

全连接结合左、右外连接的结果,使用NULL值作为两侧缺失匹配结果,示意图如下:

要注意的是MySQL中没有full join全连接关键词,要用union结合结果,使用示例如下:\

sql 复制代码
-- 左连接:所有学生 + 匹配的班级
SELECT s.*, c.* 
-- student 别名起为s
FROM student s
LEFT JOIN class c ON s.class_id = c.class_id
UNION
-- 右连接:所有班级 + 匹配的学生(补充左连接中无学生的班级)
SELECT s.*, c.* 
FROM student s
RIGHT JOIN class c ON s.class_id = c.class_id;

形成结果如下:

补充:MySQL中的UNION用于连接两个以上的 SELECT 语句的结果组合到一个结果集合,并去除重复的行,UNION ALL则会保留重复行。

交叉查询

用于生成两张表的笛卡尔积(需要全量组合的场景),左表记录记录与右表每一行的组合,示例语句如下:

sql 复制代码
SELECT *
FROM student
CROSS JOIN class;

形成结果就是内连查询不指定关联条件on时的结果:

生产环境中谨慎使用,生成大量数据,易撑爆服务器。

总结

  • 内连接(INNER JOIN):仅返回两张表满足关联条件的交集,无匹配的行直接过滤;
  • 外连接(LEFT/RIGHT/FULL):以某张表为基准保留全部数据,无匹配的字段填充 NULL;
  • 交叉查询(CROSS JOIN):无关联条件,返回两张表的笛卡尔积(全量组合);

使用方法为:

sql 复制代码
select * from table_A (left/right) join table_B on 关联条件 where 限定条件;
相关推荐
这个DBA有点耶10 小时前
NULL不是空——数据库里最反直觉的设计,90%新人踩过的坑
数据库·mysql·代码规范
这个DBA有点耶12 小时前
AI写的SQL跑崩了生产库,这锅谁背?
数据库·人工智能·程序员
镜舟科技12 小时前
Databricks 再提 LTAP,AI 时代的数据底座为何重回大一统叙事?
数据库·架构·agent
Databend13 小时前
从湖仓升级为 Agent 时代的数据控制面,Snowflake 和 Databricks 有哪些布局
大数据·数据库·agent
ClouGence16 小时前
SQL Server CDC 能放到 Always On 备库读吗?一文讲透原理与实践
数据库·sql server
先吃饱再说1 天前
存储的进化:从 MySQL 到浏览器缓存,数据到底住在哪?
数据库
Nturmoils1 天前
字段太多看不全,ksql 的展开模式和输出控制怎么用
数据库·后端
Databend2 天前
Agent 轨迹分析与归因的数据工程实践
大数据·数据库·agent
这个DBA有点耶2 天前
SQL改写进阶:标量子查询的“隐形代价”与消除实战
数据库·mysql·架构
smallyoung2 天前
数据库乐观锁深度解析:MySQL、PostgreSQL 实战 + Spring Boot 集成指南
数据库·mysql·postgresql