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

文章目录

  • 连接查询
    • 驱动表
    • 连接查询分类
  • 内连接(INNER JOIN)
    • 内连接 —— 等值连接
    • 内连接 —— 自然连接(NATURAL JOIN)
    • 内连接 —— 交叉连接(笛卡尔积)
  • 外连接(OUTER JOIN)
    • 外连接 —— 左连接(LEFT JOIN) / 左外连接(LEFT OUTER JOIN)
      • 拓展:左连接不包含内连接
    • 外连接 —— 右连接(RIGHT JOIN) / 右外连接(RIGHT OUTER JOIN)
      • 拓展:右连接不包含内连接
    • 外连接 —— 全连接 / 全外连接
      • 【注意】:FULL 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 关键字异常讨论

网上部分教程出现的 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://www.mzph.cn/pingmian/7503.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

nodejs里面的 http 模块介绍和使用

Node.js的HTTP模块是一个核心模块,它提供了很多功能来创建HTTP服务器和发送HTTP请求。 http.Server是一个基于事件的http服务器,内部是由c实现的,接口是由JavaScript封装。 http.request是一个http客户端工具。 用户向服务器发送数据。 创…

嵌入式RTOS面试题目

用过哪些嵌入式操作系统?使⽤RTOS和裸机代码开发有什么区别(优缺点)? 之前的⼀个项⽬是采⽤裸机代码开发的,写起来还⾏,通过状态机来管理业务逻辑和各种外设。 但是随着外设的增加,任务之间的…

CSDN我的创作纪念日128天||不忘初心|努力上进|勇往直前

机缘 Hello,大家好,我是景天,其实很早之前我就加入到了CSND的大军,但那是我还是个小白,经常回来CSND汲取养料,就这样慢慢的来提升自己,强大自己。经过多年的学习,积累与总结&#x…

Davinci工程开发方法论

基本概念 Flash Driver是下载到RAM里面的bin文件 Boot Manager是ROM上启动运行的第一个实例,可以是独立的bin文件,可以是集成在FBL里面。 Bootloader存储在ROM里面的bin文件 Demo Appl一个示例模板,用来跳转到Bootloader的,也是一…

本地的git仓库和远程仓库

文章目录 1. 远程创建仓库2. 关联远程和本地代码3. 推送本地分支到远程 1. 远程创建仓库 2. 关联远程和本地代码 上面创建完后会得到一个git仓库的链接,有SSH或者http的 http://gitlab.xxxxx.local:18080/xxxxx/dvr_avm.git ssh://gitgitlab.xxxxx.local:10022/xx…

220V转18V500mA非隔离恒压WT5113

220V转18V500mA非隔离恒压WT5113 亲爱的朋友们,你们是否在为如何提高电源方案而烦恼呢?今天我给大家带来了一款芯片,WT5113宽输出范围非隔离交直流转换芯片,它可是电源方案中的得力助手哦! 这款芯片拥有220V降12V、2…

Mujoco210和Mujoco-py在 Ubuntu22.04 下的安装

mujoco和mujoco-py的关系:mujoco是一个物理引擎,主要应用于强化学习和最优化控制领域。mujoco-py是mujoco编程的 Python 接口,由OpenAI Gym开发,可以使用mujoco_py方便地调用mujoco的API。 mujoco官网: https://mujoco…

国产中间件概述

1、什么是中间件? 1.1、基本概念 定义:中间件,英文名为 Middleware,是独立的系统级软件,连接操作系统层和应用程序层,将不同操作系统提供应用的接口标准化、协议统一化,屏蔽具体操作的细节。位置:居于各类应用与操作系统之间,在操作系统、网络和数据库之上,应用软件…

Linux(openEuler、CentOS8)企业内网主备DNS服务器搭建

本实验环境为openEuler系统<以server方式安装>&#xff08;CentOS类似&#xff0c;可参考本文&#xff09; 知识点 什么是DNS&#xff1a;DNS服务器&#xff0c;即域名服务器&#xff08;Domain Name Server&#xff09;&#xff0c;是进行域名和与之相对应的IP地址转换…

24年最新AI数字人简单混剪

24年最新AI数字人简单混剪 网盘自动获取 链接&#xff1a;https://pan.baidu.com/s/1lpzKPim76qettahxvxtjaQ?pwd0b8x 提取码&#xff1a;0b8x

2024年如何下载51cto的视频

你是否对学习新的技术知识感到困惑&#xff1f;是否在寻找一个优质的在线学习平台&#xff1f;那么&#xff0c;我告诉你一个秘密——51cto视频平台&#xff01;它是一个专注于技术领域的视频平台&#xff0c;提供了大量的高质量视频资源。如果你想在2024年下载51cto的视频&…

白酒:白酒香型与历史文化传承的关联性研究

云仓酒庄的豪迈白酒作为中国白酒的品牌&#xff0c;其香型与历史文化传承之间存在着紧密的关联。白酒的香型不仅是酿造技艺的体现&#xff0c;更是中华历史文化的重要组成部分。 首先&#xff0c;白酒的香型与中国的地理、气候和自然资源密切相关。不同地区的白酒产区&#xff…

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

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

QT+多线程编程

QT的多线程编程有两种 1、自定义类继承QThread 第一种是自定义一个类继承于QThread&#xff0c;重写run()方法来实现。然后当需要使用线程的时候你就新建一个自定义对象&#xff0c;然后调用start方法开始运行。 下面的例子是widget里面创建一个线程&#xff0c;然后调用sta…

BGP协议应用:SW1、SW2、SW3、RT1、RT2之间运行BGP协议

8.SW1、SW2、SW3、RT1、RT2之间运行BGP协议,SW1、SW2、RT1 AS号65001、RT2 AS号65002、SW3 AS号65003。 (1)SW1、SW2、SW3、RT1、RT2之间通过Loopback1建立IPv4 BGP邻居。SW1和SW2之间财务通过Loopback2建立IPv4 BGP邻居,SW1和SW2的Loopback2互通采用静态路由。 (2)SW1…

C++基础——深拷贝和浅拷贝

C中类的拷贝有两种&#xff1a;深拷贝&#xff0c;浅拷贝&#xff1a;当出现类的等号赋值时&#xff0c;即会调用拷贝函数 一、概念 浅拷贝&#xff1a;同一类型的对象之间可以赋值&#xff0c;使得两个对象的成员变量的值相同&#xff0c;两个对象仍然是独立的两个对象&#…

Centos上通过Docker安装单节点ES

系列文章目录 文章目录 系列文章目录前言 前言 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到网站&#xff0c;这篇文章男女通用&#xff0c;看懂了就去分享给你的码吧。 Elasticsearch 是一…

美易官方:美股周一收高,道指连续第四个交易日上涨

收盘之际&#xff0c;美股市场周一的表现可圈可点&#xff0c;各大股指纷纷走高&#xff0c;道指更是连续第四个交易日实现上涨。这一积极态势不仅凸显了投资者对于全球经济的信心&#xff0c;也反映了市场对于未来前景的乐观预期。 道指涨176.59点&#xff0c;涨幅为0.46%&…

最新贷款市场报价利率(LPR)数据(1991-2024)

数据来源&#xff1a;东方财富网时间跨度&#xff1a;1991-2024年 数据范围&#xff1a;全国范围 数据指标&#xff1a; LPR_1Y利率(%) LPR_5Y利率(%) 中长期贷款利率:5年以上(%) 短期贷款利率:6个月至1年(含)(%) 日期 样例数据&#xff1a; 下载链接&#xff1a; ht…

枚举类型和联合体类型

一、 枚举类型 简单介绍一下 枚举 顾名思义 就是一一例举 像我们生活中的很多东西 比如说一周有多少天 性别有几种 这些都可以用枚举类型来一一列举 1. 枚举类型的定义 我们以性别为例子 这里如果我们开始打印它们的数值的话 我们会发现 typedef enum Sex {MALE,FEMALE…