【MySQL】连接查询(JOIN 关键字)—— 图文详解:内连接、外连接、左连接、左外连接、右连接、右外连接

文章目录

  • 连接查询
    • 驱动表
    • 连接查询分类
  • 内连接(INNER JOIN)
    • 内连接 —— 等值连接
    • 内连接 —— 自然连接(NATURAL JOIN)
    • 内连接 —— 交叉连接(笛卡尔积)
  • 外连接(OUTER JOIN)
    • 外连接 —— 左连接(LEFT JOIN) / 左外连接(LEFT OUTER JOIN)
      • 拓展:左连接不包含内连接
    • 外连接 —— 右连接(RIGHT JOIN) / 右外连接(RIGHT OUTER JOIN)
      • 拓展:右连接不包含内连接
    • 外连接 —— 全连接 / 全外连接
      • 拓展:全连接不包括内连接

我是一名立志把细节都说清楚的博主,欢迎【关注】🎉 ~

原创不易, 如果有帮助 ,记得【点赞】【收藏】 哦~ ❥(^_-)~

如有错误、疑惑,欢迎【评论】指正探讨,我会尽可能第一时间回复的,谢谢支持


连接查询

MySQL使用 JOIN 关键字连接多个表查询数据,主要使用的是嵌套循环连接算法(nested-loop join)。这种算法机制简单的理解可以类比为for循环遍历一样。

从驱动表中选取数据作为循环的基础数据,然后以这些数据作为 查询条件 到下一个表中进行循环遍历查询。循环往复。

这种算法的缺点是: 连接的表越多,循环嵌套的层数就越多,算法复杂度呈指数级增长。

对应的处理方法是: 我们在设计查询时,尽可能减少连接表的个数。

驱动表

驱动表: 在使用多表嵌套连接时,首先,会全表扫描该一个表作为查询条件,这个表叫做驱动表。然后用驱动表返回的结果集逐行去匹配的表,叫做被驱动表。

关于驱动表的详细说明及性能优化,本文不过多提及,感兴趣的可以看这篇文章:
【MySQL】驱动表、被驱动表详解。—— 性能优化。

连接查询分类

  • 内连接(INNER JOIN)
    • 等值连接(最常用)
    • 自然连接
    • 交叉连接
  • 外连接(INNER JOIN)
    • 左连接(LEFT JOIN) / 左外连接(LEFT OUTER JOIN)
    • 右连接(RIGHT JOIN) / 有外连接(RIGHT OUTER JOIN)
  • 全连接

内连接(INNER JOIN)

内连接查询的是两张表的交集,即两张表都有的数据。

内连接的驱动表:通常是数据量较少的表作为驱动表。

内连接概念图


内连接 —— 等值连接

SQL 代码形式如下:

# FROM 两表 WHERE 连接。
SELECT * FROM A表 , B表 WHERE A表.id = B表.id;# JOIN ON 连接。
SELECT * FROM A表  JOIN B表 ON A表.id = B表.id;# INNER JOIN ON 连接。
SELECT * FROM A表 INNER JOIN B表 ON A表.id = B表.id;# 多JOIN ON 连接。
SELECT * FROM A表 
INNER JOIN B表 ON A表.id = B表.id
INNER JOIN C表 ON B表.id = C表.id;

内连接 —— 自然连接(NATURAL JOIN)

自然连接会在两个表中寻找那些名称相同的列,并且以这些列的值作为联接的条件。不需要我们指定连接条件。

案例说明

假设有两个表,一个是员工表 employees 和一个是部门表 departments。

employees 表:

+--------+------+-------------+--------+
| emp_id | name |  department | salary |
+--------+------+-------------+--------+
|      1 | John |       Sales |  50000 |
|      2 | Mary | Engineering |  60000 |
|      3 |  Bob |       Sales |  40000 |
+--------+------+-------------+--------+

departments 表:

+---------+-------------+
| dept_id |  department |
+---------+-------------+
|       1 |       Sales |
|       2 | Engineering |
|       3 |  Accounting |
+---------+-------------+

如果我们想要找到所有员工及其部门的信息,我们可以使用自然连接来联接这两个表:

SELECT *
FROM employees
NATURAL JOIN departments;

这将返回两个表中同名的列 department 的值相等的那些行,因此结果集将是:

+--------+------+-------------+--------+---------+-------------+
| emp_id | name |  department | salary | dept_id |  department |
+--------+------+-------------+--------+---------+-------------+
|      1 | John |       Sales |  50000 |       1 |       Sales |
|      2 | Mary | Engineering |  60000 |       2 | Engineering |
+--------+------+-------------+--------+---------+-------------+

注意,在这个例子中,emp_id 和 dept_id 是自然连接过程中被删除的重复列。

自然连接特点:

  • 如果两个表中有同名的非空列,并且列中的值相等,那么这一行会出现在结果集中。
  • 如果两个表中有同名的列,但是列中的值不相等,那么这一行不会出现在结果集中。
  • 如果两个表中有同名的列,但是至少有一个列是空的,那么这一行也不会出现在结果集中。
  • 自然连接会删除重复的列,只保留一个同名的列。

因为自然连接的特点,我们无法指定连接列,这种写法给给我们带来了不确定性。如果以后数据模型变更导致原来可以自然连接的列,不能再自然连接了,导致数据查询异常。所以通常情况下,不推荐使用自然连接


内连接 —— 交叉连接(笛卡尔积)

由没有联结条件的表关系返回的结果叫笛卡儿积。检索出的行的数目将是第一个表中的行数乘以第二个表中的行数。

# 笛卡尔积示例一:
SELECT * FROM A表 JOIN B表;# 笛卡尔积示例二:
SELECT * FROM A表, B表;

我是一名立志把细节都说清楚的博主,欢迎【关注】🎉 ~

原创不易, 如果有帮助 ,记得【点赞】【收藏】 哦~ ❥(^_-)~

如有错误、疑惑 ,欢迎【评论】指正探讨,我会尽可能第一时间回复的,谢谢支持


外连接(OUTER JOIN)

外连接可以保留连接表所有的记录,包括这条记录没有匹配的记录也可以保留(以NULL形式出现)。

保留表可以根据保留左表、右表、全表从而分为:

  • 左连接/左外连接。
  • 右连接/右外连接。
  • 全连接/全外连接。

左表、右表的判断标准,是以包含 JOIN 的关键字作为基准(如:LEFT JOINLEFT OUTER JOINRIGHT JOINRIGHT OUT JOIN):

  • 关键字的左边称为左表。
  • 关键字的右边称为右表。

外连接 —— 左连接(LEFT JOIN) / 左外连接(LEFT OUTER JOIN)

左连接 / 左外连接

左连接会返回左表中的所有记录。如果右表中没有匹配的记录,则右表字段使用 NULL 填充。

# LEFT JOIN ON 连接。
SELECT * FROM A表  
LEFT JOIN B表 
ON A表.id = B表.id;# RIGHT JOIN ON 连接。
SELECT * FROM A表  
RIGHT JOIN B表 
ON A表.id = B表.id;

Employees 表:

+----+----------+
| id | name     |
+----+----------+
| 1  | Alice    |
| 7  | Bob      |
| 11 | Meir     |
| 90 | Winston  |
| 3  | Jonathan |
+----+----------+

EmployeeUNI 表:

+----+-----------+
| id | unique_id |
+----+-----------+
| 3  | 1         |
| 11 | 2         |
| 90 | 3         |
+----+-----------+

左连接SQL样例:

SELECT unique_id, name
FROMEmployees
LEFT JOINEmployeeUNI
ONEmployees.id = EmployeeUNI.id;

结果:

+-----------+----------+
| unique_id | name     |
+-----------+----------+
| null      | Alice    |
| null      | Bob      |
| 2         | Meir     |
| 3         | Winston  |
| 1         | Jonathan |
+-----------+----------+

拓展:左连接不包含内连接

左连接不包含内连接

模版样式:

SELECT * FROM A表  
LEFT JOIN B表 
ON A表.id = B表.id
WHERE B表.id IS NULL;

外连接 —— 右连接(RIGHT JOIN) / 右外连接(RIGHT OUTER JOIN)

右连接(RIGHT JOIN) / 右外连接(RIGHT OUTER JOIN)
与左连接同理。
右连接会返回右表中的所有记录。如果左表中没有匹配的记录,则左表字段使用 NULL 填充

通常情况下,右连接会被习惯性的改写成左连接。效果是一样的,左连接的可读性更好点。

模版样式:

SELECT * FROM A表  
RIGHT JOIN B表 
ON A表.id = B表.id;

拓展:右连接不包含内连接

右连接不包含内连接

模版样式:

SELECT * FROM A表  
RIGHT JOIN B表 
ON A表.id = B表.id
WHERE A表.id IS NULL;

外连接 —— 全连接 / 全外连接

全连接 / 全外连接

全连接(也称全外连接)一般没有什么意义,MySQL并不直接支持全外连接,但可以通过左右外连接的并集(UNION 关键字)来模拟实现。

SELECT * FROM A表 LEFT JOIN B表 ON A表.id = B表.id
UNION
SELECT * FROM A表 RIGHT JOIN B表 ON A表.id = B表.id

注意:
网上部分教程出现的 FULL OUTER JOIN 关键词,但是MySQL并不直接支持全外连接,所以可能很多本部 无法直接使用 FULL OUTER JOIN 关键词,至少我测试的版本是这样。所以在使用前建议测试一下。

# 错误SQL演示
SELECT *
FROM  A表FULL OUTER JOIN B表
ON A表.id = B表.id

为此我做了测试,我这里使用的 MySQL 8.0.32 版本,并不识别 FULL OUTER JOIN 关键词。并且执行搜索提示错误。

错误演示


拓展:全连接不包括内连接

全连接不包括内连接

SELECT * FROM A表 LEFT JOIN B表 ON A表.id = B表.id
UNION
SELECT * FROM A表 RIGHT JOIN B表 ON A表.id = B表.id
WHERE A表.id IS NULLOR B表.id IS NULL

我是一名立志把细节都说清楚的博主,欢迎【关注】🎉 ~

原创不易, 如果有帮助 ,记得【点赞】【收藏】 哦~ ❥(^_-)~

如有错误、疑惑 ,欢迎【评论】指正探讨,我会尽可能第一时间回复的,谢谢支持

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://xiahunao.cn/news/3014865.html

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

专注 APT 攻击与防御——红蓝对抗渗透测试

在团体渗透测试的项目中,如红蓝对抗,团队渗透测试比赛等,最重要的是过程与结果实时共享于团队,例如:A同学nmap目标站,B同学也nmap目标站,这在对抗比赛中是极其浪费时间也是非常容易引起防火墙&a…

IDEA无法下载远程仓库jar包问题

问题描述: idea无法下载远程仓库jar包,最奇怪的是idea有多个项目,有些项目可以下载,有些项目不行。报错如下: 一开始: unable to find valid certification path to requested target Try run Maven impo…

SQLI-labs-第十三关和第十四关

目录 第十三关 1、判断注入点 2、判断当前数据库 3、爆表名 4、爆字段名 5、爆值 第十四关 1、判断注入点 知识点:POST方式的单引号和括号闭合错误,报错注入 第十三关 思路: 1、判断注入点 使用Burpsuite抓包 首先加入一个单引号,…

小米手机短信删除了怎么恢复?这里教你快速解决!

手机已经成为我们生活中不可或缺的一部分,比如小米手机。我们通过手机进行通讯、娱乐、学习等各种活动,其中,短信是我们日常生活中的重要信息来源之一。然而,我们可能会不小心删除了一些重要的短信,这时候我们就会想知…

【JS】call和 apply函数的详解

JavaScript 中 call() 和 apply() 函数的详解 在JavaScript中,call()和apply()都是非常重要的方法,用于调用函数时指定函数体内的this的值,从而实现不同对象之间的方法共享。尽管它们的功能非常相似,但在实际使用中各有其优势和特…

【前端】实现表格简单操作

简言 表格合并基础篇 本篇是在上一章的基础上实现,实现了的功能有添加行、删除行、逆向选区、取消合并功能。 功能实现 添加行 添加行分为在上面添加和在下面追加行。 利用 insertAdjacentElement 方法实现,该方法可以实现从前插入元素和从后插入元…

两个开关控制一盏灯

1.PLC控制电路 1.1双联控制 开关控制灯最为普遍、简单的做法是用一个开关控制一盏灯,除此之外,在某个位置需要用两个开关控制一盏灯,这就是双联控制。 灯的双联控制,在传统上要用到2个单刀双联开关,其电气原理图如下图…

【CTS :testExtensionAvailability】

【CTS】android.hardware.camera2.cts.CameraExtensionCharacteristicsTest#testExtensionAvailability 报错: java.lang.AssertionError: Extensions system property : true does not match with the advertised extensions: false expected: but was: 通过对这…

暗区突围端游海外版预下载教程 暗区突围端游海外版怎么注册 下载

暗区突围端游海外版预下载教程 暗区突围端游海外版怎么注册 下载 想必最近暗区突围PC版本的上线对于热爱这款游戏的玩家们是一件喜事,这款游戏自从手游上线之初就在全世界范围内引起了不小的轰动,作为逃离塔科夫这款游戏的竞品,刚上线时自然…

嵌入式学习——C语言基础——day14

1. 共用体 1.1 定义 union 共用名 { 数据类型1 成员变量1; 数据类型2 成员变量2; 数据类型3 成员变量3; .. }; 1.2 共用体和结构体的区别 1. 结构体每个成员变量空间独立 2. 共用体每个成员变量空间共享 1.3 判断内存大小端 1. 内存大端…

doris 启动be报错

doris版本是1.2.4 java版本是:1.8 刚开始我以为是版本不兼容问题,后面发现思路错了,版本是兼容的,报以下错我的原因是操作系统没有达到安装要求 以下是博主在部署doris x64(avx2)版本中遇到的小bug 在大家使用doris的时候应该…

翻译《The Old New Thing》- Does Windows have a limit of 2000 threads per process?

Does Windows have a limit of 2000 threads per process? - The Old New Thing (microsoft.com)https://devblogs.microsoft.com/oldnewthing/20050729-14/?p34773 Raymond Chen 2005年07月29日 Windows 是否有一个每个进程2000线程的限制? 简要 文章解释了在 W…

javascript中的DOM和BOM

目录 JavaScript中的对象 简介: js对象的基本用法: 创建对象: 访问对象的属性: 设置修改对象的属性: 删除对象的属性: DOM(文档对象模型) 简介: DOM对象的属性…

探索未来居住新纪元:公寓商场综合楼可视化引领潮流

在繁忙的都市生活中,我们都渴望找到一处既能满足居住需求,又能提供便捷购物体验的理想之地。如今,这一梦想已不再是遥不可及,随着科技的飞速发展,3D可视化技术正逐渐走进我们的生活,为我们带来前所未有的居…

用脚本写一个日期样式的字符

现在想要诸如此类样式的语句:(过去三个月的) 可以用python脚本写: from datetime import date, timedelta# 获取当前日期 current_date date.today()# 定义过去三个月的时间间隔 three_months_ago current_date - timedelta(da…

第二证券|集合竞价915到920能成交吗?

不能,买卖日9点15分到9点20分是集合竞价申报时刻,还没有发生集合竞价。 在这个时刻段内,投资者能够申报,也能够撤单,但这些挂单并不会立即成交。9:25时,系统会对一切收到的挂单进行一次性促成处理&#xf…

python 12实验

1.导入数据。 2.清洗数据,将缺失值或“NAN”替换为“无”,并将文本数据转换为数值型数据。 3.使用聚类算法(如KMeans)对数据进行聚类,并计算样本到簇中心的平均距离以确定最佳的簇数量。 4.对数据进行PCA降维&#xff…

酷开科技AI技术支持,酷开系统根据你的喜好量身定制节目

在当今数字化时代,个性化推荐已成为提升消费者体验的关键因素。酷开科技的智慧AI,为消费者提供了精彩的内容推荐服务,更大地丰富了消费者的娱乐生活。 酷开系统中的AI推荐引擎通过学习消费者的观看习惯和偏好,能够快速识别其兴趣…

「51媒体」企业单位新闻稿件考核,怎么发布

传媒如春雨,润物细无声,大家好,我是51媒体网胡老师。 电力税务企事业单位部门等单位提供了新闻稿件,如何在一些重点媒体进行宣发呢: 精准锁定发布媒体 了解考核要求:仔细阅读宣传任务名单,了解…

Leetcode167两数之和

题目链接&#xff1a; 167两数之和 解题思路: 缩减空间法 // 167 两数之和 缩减搜索空间方法 vector<int> twoSum(vector<int>& numbers, int target) {int i 0;int j numbers.size() - 1;while (i < j){int tmp numbers[i] numbers[j];if (tmp tar…