当前位置: 首页 > news >正文

[MySQL初阶]MySQL(7) 表的内外连接

标题:[MySQL初阶]MySQL(7)表的内外连接
@水墨不写bug


在这里插入图片描述


文章目录

  • 一. 内连接 (INNER JOIN)
  • 二. 外连接 (OUTER JOIN)
    • 关键区别总结
  • 三、 如何选择


在 MySQL 中,连接(JOIN)用于根据两个或多个表之间的相关列组合行。内连接(INNER JOIN)和外连接(OUTER JOIN)的核心区别在于它们如何处理连接条件不匹配的行。

一. 内连接 (INNER JOIN)

  • 作用: 返回两个表中连接条件都匹配的行。
  • 结果集: 仅包含满足连接条件的行。如果一个表中的行在另一个表中没有匹配项,则该行不会出现在结果中。
  • NULL 值: 结果集中不会出现 NULL 值(因为只返回匹配成功的行)。
  • 语法:
SELECT ...
FROM table1
[INNER] JOIN table2 ON table1.column = table2.column;
-- `INNER` 关键字通常可以省略

内连接相当于之前写的两表做笛卡尔积后再通过where子句选择,意味着下面的这两句SQL是等价的:

#用之前的写法
select * from emp,dept where emp.deptno=dept.deptno and ename='lisi';
#用标准的内连接写法
select * from emp inner join dept on emp.deptno=dept.deptno and ename='lisi';

图示:

二. 外连接 (OUTER JOIN)

外连接强制保留一个或两个表的所有行,即使它们在另一个表中没有匹配项。根据保留哪个表的数据,外连接分为三种:

  • a. 左外连接 (LEFT [OUTER] JOIN)
    • 作用: 返回左表的所有行,以及右表中满足连接条件的匹配行。如果右表没有匹配行,则右表列用 NULL 填充。
    • 结果集: 左表所有行 + 匹配的右表行(无匹配则右表部分为 NULL)。
    • NULL 值: 当右表没有匹配行时,结果集中右表对应的列会显示为 NULL。
    • 语法:
SELECT ...
FROM table1
LEFT [OUTER] JOIN table2 ON table1.column = table2.column;

图示:

Table A (左)     Table B (右)
+----+----+     +----+----+
| id | val|     | id | val|
+----+----+     +----+----+
| 1  | A  |     | 2  | X  |
| 2  | B  |     | 3  | Y  |
| 3  | C  |     | 4  | Z  |
+----+----+     +----+----+LEFT JOIN ON A.id = B.id:
+----+----+------+------+
| id | val| id   | val  |
+----+----+------+------+
| 1  | A  | NULL | NULL | <-- A 表行 1 保留,B 无匹配
| 2  | B  | 2    | X    |
| 3  | C  | 3    | Y    |
+----+----+------+------+

左外连接含义:返回左表(A)所有的记录,即使右表(B)中没有匹配的记录,B 表无匹配时结果为 NULL。
保留了左表 A 的所有行(id 1, 2, 3)。id 1 在 B 中无匹配,B 的列显示为 NULL。

  • b. 右外连接 (RIGHT [OUTER] JOIN)
    • 作用: 返回右表的所有行,以及左表中满足连接条件的匹配行。如果左表没有匹配行,则左表列用 NULL 填充。
    • 结果集: 右表所有行 + 匹配的左表行(无匹配则左表部分为 NULL)。
    • NULL 值: 当左表没有匹配行时,结果集中左表对应的列会显示为 NULL。
    • 语法:
SELECT ...
FROM table1
RIGHT [OUTER] JOIN table2 ON table1.column = table2.column;

图示:

Table A (左)     Table B (右)
+----+----+     +----+----+
| id | val|     | id | val|
+----+----+     +----+----+
| 1  | A  |     | 2  | X  |
| 2  | B  |     | 3  | Y  |
| 3  | C  |     | 4  | Z  |
+----+----+     +----+----+RIGHT JOIN ON A.id = B.id:
+------+------+----+----+
| id   | val  | id | val|
+------+------+----+----+
| 2    | B    | 2  | X  |
| 3    | C    | 3  | Y  |
| NULL | NULL | 4  | Z  | <-- B 表行 4 保留,A 无匹配
+------+------+----+----+

右外连接含义:返回右表(B)所有的记录,即使左表(A)中没有匹配的记录,A 表无匹配时结果为 NULL。
保留了右表 B 的所有行(id 2, 3, 4)。id 4 在 A 中无匹配,A 的列显示为 NULL。

  • c. 全外连接 (FULL [OUTER] JOIN)
    • 作用: 返回左表和右表的所有行。只要其中一个表有匹配的行,就返回该行。 当某行在另一个表中没有匹配项时,另一个表的列用 NULL 填充。
    • 结果集: 左表所有行 + 右表所有行(无匹配的部分用 NULL 填充)。
    • NULL 值: 当左表或右表没有匹配行时,结果集中对应另一方的列会显示为 NULL。
    • 语法: MySQL 不直接支持 FULL OUTER JOIN。通常通过组合 LEFT JOINRIGHT JOIN 并使用 UNION(或 UNION ALL)来模拟实现:
SELECT ...
FROM table1
LEFT JOIN table2 ON table1.column = table2.column
UNI0N [ALL] -- 使用 UNI0N ALL 会保留重复行(通常不应该有),UNI0N 会去重
SELECT ...
FROM table1
RIGHT JOIN table2 ON table1.column = table2.column
WHERE table1.column IS NULL; -- 这个 WHERE 条件是为了只取 RIGHT JOIN 中左表为 NULL 的部分,避免重复

图示:

Table A         Table B
+----+----+     +----+----+
| id | val|     | id | val|
+----+----+     +----+----+
| 1  | A  |     | 2  | X  |
| 2  | B  |     | 3  | Y  |
| 3  | C  |     | 4  | Z  |
+----+----+     +----+----+FULL OUTER JOIN ON A.id = B.id:
+------+------+------+------+
| id   | val  | id   | val  |
+------+------+------+------+
| 1    | A    | NULL | NULL | <-- 仅存在于 A
| 2    | B    | 2    | X    |
| 3    | C    | 3    | Y    |
| NULL | NULL | 4    | Z    | <-- 仅存在于 B
+------+------+------+------+

全外连接含义:返回左右两表所有的记录,任何一方无匹配时用 NULL 填充。MySQL 需用 UNION 实现。

关键区别总结

特性内连接 (INNER JOIN)左外连接 (LEFT JOIN)右外连接 (RIGHT JOIN)全外连接 (FULL JOIN)
匹配行返回返回返回返回
左表不匹配行不返回返回 (NULL填充)不返回返回 (NULL填充)
右表不匹配行不返回不返回返回 (NULL填充)返回 (NULL填充)
结果来源仅交集左表全集 + 匹配的右表右表全集 + 匹配的左表左表全集 + 右表全集
NULL 值不产生在右表列产生在左表列产生在左表或右表列产生

三、 如何选择

  • 需要两边都匹配的数据时:INNER JOIN(最常见)。
  • 需要主表所有记录 + 关联表匹配信息(即使关联表没有匹配项)时:
    • 主表在 FROM 后 / 第一个 JOIN 前 -> 用 LEFT JOIN
    • 主表在 JOIN 关键字后 -> 用 RIGHT JOIN(相对较少用,通常通过调整表顺序改用 LEFT JOIN)。
  • 需要两个表所有记录时:FULL OUTER JOIN(MySQL 需模拟实现)。

示例场景:

  • INNER JOIN: 查询所有下过订单的客户及其订单信息。
  • LEFT JOIN: 查询所有客户及其订单信息(包括从未下过订单的客户)。
  • RIGHT JOIN: 查询所有订单及其对应的客户信息(包括异常订单,其客户ID可能无效或被删除)。通常用 LEFT JOIN 并调换表顺序实现。
  • FULL JOIN: 查询所有客户和所有订单(列出所有客户,无论是否有订单;列出所有订单,无论客户信息是否有效)。

完~
未经作者同意禁止转载
在这里插入图片描述

http://www.lqws.cn/news/70021.html

相关文章:

  • 什么是线程上下文切换?
  • Linux(10)——第二个小程序(自制shell)
  • 摩尔投票算法原理实现一文剖析
  • SQL Views(视图)
  • MyBatis源码解析:从 Mapper 接口到 SQL 执行的完整链路
  • 数据库系统概论(十二)SQL 基于派生表的查询 超详细讲解(附带例题表格对比带你一步步掌握)
  • Spring Boot中的WebSocket技术实现
  • oracle sql 语句 优化方法
  • NLP学习路线图(十九):GloVe
  • Spring Boot中保存前端上传的图片
  • Android高级开发第三篇 - JNI异常处理与线程安全编程
  • 使用 PHP 和 Guzzle 对接印度股票数据源API
  • 【Android】MT6835 + MT6631 WiFi进入Meta模式出现WiFi_HQA_OpenAdapter failed
  • 【Elasticsearch】Elasticsearch 核心技术(一):索引
  • 大数据-275 Spark MLib - 基础介绍 机器学习算法 集成学习 随机森铃 Bagging Boosting
  • MySQL数据库从0到1
  • 基于FPGA的VGA显示文字和动态数字基础例程,进而动态显示数据,类似温湿度等
  • 监控 100 台服务器磁盘内存CPU利用率
  • MPTCP 聚合吞吐
  • StarRocks部署方案详解:从单机到分布式集群
  • TDengine 的 AI 应用实战——电力需求预测
  • 【存储基础】存储设备和服务器的关系和区别
  • 第1篇:数据库中间件概述:架构演进、典型方案与应用场景
  • 序列搜索策略
  • PromQL 快速上手
  • 4.RV1126-OPENCV 图像轮廓识别
  • 【SpringBoot】| 接口架构风格—RESTful
  • 什么是 CPU 缓存模型?
  • 深入探讨redis:缓存
  • vue-13(延迟加载路由)