MySQL 中的各种 JOIN 本文主要介绍 SQL 标准中定义的各种连接的意义和区别,例如 , 交叉连接( CROSS JOIN )、内连接( INNER JOIN )、外连接( OUTER JOIN )、自然连接( NATURAL JOIN )等,并结合例子讲解这些连接在 MySQL 中的语法和表现 。
从网上的资料看,JOIN更多翻译为连接,本文中凡是出现中文“连接”的地方都是指JOIN。
本文中用到的所有例子涉及两张表——customers用户表和orders订单表,其中订单表中的cust_id字段表示用户的唯一 ID,也就是用户表的主键cust_id。两张表的数据如下:
注:两张表都经过了简化,实际业务中这两张表肯定还包括其他字段 。
英文维基百科JOIN词条 对连接的定义如下:
翻译过来就是 , “连接可以根据一张(自连接)或多张表中的共同值将这些表的列数据合并为一个新的结果集,标准 SQL 定义了五种连接:内连接、左外连接、右外连接、全外连接和交叉连接 。”
也就是说,连接是 SQL 标准中定义的一种组合多张表的方式,当然一张表自身也可以和自身组合,称为自连接 。连接后得到的结果集的每一列其实都来自用于连接的多张表 , 不同的连接类型只是区分了这些列具体从哪张表里来,列里填充的是什么数据 。
【mysql怎么表示外连接 mysql外连接和内连接】 其实英文维基百科的JOIN词条已经把各种连接的类型解释地非常清楚了,非常值得去看一下 。
我们来看一下 SQL 标准中定义的各种连接类型 , 理解各种连接最好的方法就是把需要连接的表想象成集合 , 并画出可以反映集合的交与并的情况的图——韦恩图,例如下图就画出了 SQL 中定义的几种主要连接 。
请先仔细查看一下图中的内容,你可以从中归纳出几种连接类型呢?
虽然图中画了 7 种集合的交并情况,但是总结起来,主要是两种连接类型在起作用——内连接( INNER JOIN )和外连接( OUTER JOIN ),其中外连接又分为了左外连接( LEFT OUTER JOIN )、右外连接( RIGHT OUTER JOIN )和全外连接( FULL OUTER JOIN ) 。
下面先简单介绍一下 SQL 标准中各种连接的定义,然后在「MySQL 中的连接」一节再用例子来演示 MySQL 中支持的各种连接 。
连接既然是用来合并多张表的,那么要定义一个连接就必须指定需要连接的表,并指定可选的连接条件 。例如,一个典型的 SQL 连接语句如下:
我们用表 A 和表 B 指代需要连接的两张表,经过 内连接 后得到的结果集 仅 包含所有满足 连接条件 的数据;而经过 外连接 后得到的数据集 不仅 包含满足 连接条件 的数据,还包含其他数据,具体的差别是:
在上面「SQL 标准定义的主要连接」一图中并没有列出交叉连接 , 交叉连接会对连接的两张表做笛卡尔积,也就是连接后的数据集中的行是由第一张表中的每一行与第二张表中的每一行配对而成的,而不管它们 逻辑上 是否可以搭配在一起 。假设交叉连接的两张表分别有 m 和 n 行数据,那么交叉连接后的数据集就包含 m 乘以 n 行数据 。
连接根据连接的条件不同,又可以区分为等值连接和非等值连接,「SQL 标准定义的主要连接」图中画出的连接的连接条件都是比较两个字段是否相等,它们都是等值连接 。
自然连接是等值连接的一种特殊形式,自然连接会自动选取需要连接的两张表中字段名相同的 所有 列做相等比较,而不需要再指定连接条件了 。
注:以下内容全部基于 MySQL 5.7 版本 , 所有例子只保证在 MySQL 5.7 上是可以正确执行的 。
MySQL 中支持的连接类型和关键字如下:
上面的表示方法摘自 MySQL 5.7 版本 官方文档 ,其中|表示两者皆可出现, []表示的是可选的, {}表示的是必选的,例如NATURAL LEFT JOIN和NATURAL JOIN都是合法的 。
可以看到,除了全外连接( FULL OUTER JOIN )以外,MySQL 基本支持了 SQL 标准中定义的各种连接 。在 MySQL 中全外连接可以通过UNION合并的方式做到,当然前提是你知道自己为什么需要这么做,具体参见: Full Out Join in MySQL。
MySQL 语法中还支持一个并不在 SQL 标准中的STRAIGHT_JOIN ,它在 表现上 和内连接或者交叉连接并无区别,只是一种给 MySQL 优化器的一个提示,STRAIGHT_JOIN提示 MySQL 按照语句中表的顺序加载表,只有在你明确清楚 MySQL 服务器对你的JOIN语句做了负优化的时候才可能用到它 。
还有一点需要说明的是,根据 官方文档 , 在 MySQL 中 , JOIN 、 CROSS JOIN和INNER JOIN实现的功能是一致的,它们在语法上是等价的 。从语义上来说 , CROSS JOIN特指无条件的连接(没有指定ON条件的JOIN或者没有指定WHERE连接条件的多表SELECT ), INNER JOIN特指有条件的连接(指定了ON条件的JOIN或者指定了WHERE连接条件的多表SELECT ) 。当然,如果你非要写... CROSS JOIN ... ON ...这样的语法,也是可以执行的,虽然写着交叉连接,实际上执行的是内连接 。
下面我们就用例子来看一看 MySQL 中支持的几种连接的例子 。
注:下面的例子都没有指定ORDER BY子句,返回结果的顺序可能会因为数据插入顺序的不同而略有不同 。
MySQL 的交叉连接或内连接有两种写法,一种是使用JOIN并用ON或者USING 子句指定连接条件的写法,一种是普通的SELECT多表 , 并且用WHERE子句指定连接的键的写法 。
下面的例子是一个交叉连接:
上面的写法等价于:
当然,第二种写法中如果将CROSS JOIN替换成JOIN或者INNER JOIN也是可以正确执行的 。上面两条语句的执行结果如下:
可以看到共返回了 30 行结果,是两张表的笛卡尔积 。
一个内连接的例子如下:
上面的写法等价于:
在连接条件比较的字段相同的情况下,还可以改用USING关键字,上面的写法等价于:
上面三条语句的返回结果如下:
可以看到只返回了符合连接条件customers.cust_id = orders.cust_id的 6 行结果,结果的含义是所有有订单的用户和他们的订单 。
左外连接和右外连接的例子如下,其中的OUTER关键字可以省略:
其中右外连接的返回与内连接的返回是一致的(思考一下为什么) , 左外连接的返回结果如下:
可以看到一共返回了 8 行数据,其中最后两行数据对应的order_id的值为NULL,结果的含义是所有用户的订单,不管这些用户是否已经有订单存在了 。
根据前面介绍的自然连接的定义,自然连接会自动用参与连接的两张表中 字段名相同 的列做等值比较,由于例子中的customers和orders表只有一列名称相同,我们可以用自然连接的语法写一个与上面的内连接的例子表现行为一样的语句如下:
可以看到 , 使用自然连接就不能再用ON子句指定连接条件了,因为这完全是多余的 。
当然,自然连接同样支持左外连接和右外连接 。
下面用一个customers表自连接的例子再来说明一下自然连接,语句如下:
因为是自连接 , 因此必须使用AS指定别名,否则 MySQL 无法区分“两个”customers表 , 运行的结果如下:
可以看到结果集和customers表完全一致,大家可以思考一下为什么结果是这样的 。
文章之前也提到了,MySQL 还支持一种 SQL 标准中没有定义的“方言” , STRAIGHT_JOIN,STRAIGHT_JOIN支持带ON子句的内连接和不带ON子句的交叉连接 , 我们来看一个STRAIGHT_JOIN版本的内连接的例子:
返回结果与前面内连接的例子是一致的,如下:
STRAIGHT_JOIN的表现和JOIN是完全一致的 , 它只是一种给 MySQL 优化器的提示,使得 MySQL 始终按照语句中表的顺序读取表(上面的例子中,MySQL 在执行时一定会先读取customers表 , 再读取orders表),而不会做改变读取表的顺序的优化 。关于 MySQL 优化器的话题这里不做展开 , 需要说明的是除非你非常清楚你在做什么,否则不推荐直接使用STRAIGHT_JOIN。
你能理解上面的语句是在检索什么数据吗?
本文主要介绍了 SQL 标准里定义的各种连接的概念,以及 MySQL 中的实现,并通过各种例子来介绍了这些连接的区别 。这些连接不一定都能在实际开发中用到,但是做到心中有知识也还是很有必要的 。
那么,现在再回忆一下,什么是内连接、外连接、自连接、等值连接和自然连接?他们的区别是什么?
最后,给大家留一个思考题,为什么 MySQL 中没有左外连接或者右外连接版本的STRAIGHT_JOIN ?
MySQL的连接查询有几种类型,每种类型的含义是什么?分为内连接,左连接,右连接和全连接四种,内连接inner join :组合两个表中的记录 , 返回关联字段相符的记录,也就是返回两个表的交集(阴影)部分 。左连接left join 是left outer join的简写,它的全称是左外连接,是外连接中的一种 。左(外)连接,左表(a_table)的记录将会全部表示出来 , 而右表(b_table)只会显示符合搜索条件的记录 。右表记录不足的地方均为NULL 。右连接right join是right outer join的简写,它的全称是右外连接,是外连接中的一种 。与左(外)连接相反,右(外)连接,左表(a_table)只会显示符合搜索条件的记录,而右表(b_table)的记录将会全部表示出来 。左表记录不足的地方均为NULL 。全连接union 单独取出的列数必须相同
MySQL之主键、外键、连接设置主键的思路:业务字段做主键、自增字典做主键、手动赋值字段做主键 。
多表查询:分散在多个不同的表里的数据查询出来的操作 。
外键:从表中用来引用主表中数据的那个公共字段 。
连接有2种类型的连接:内连接(inner join)和外连接(outer join)
内连接:查询结果只返回符合连接条件的记录 。
外连接:查询结果返回某一个表中的所有记录,以及另一个表中满足连接条件的记录 。
其中左连接:left join,返回左表中的所有记录,以及右表中符合连接条件的记录 。
右连接:right join,返回右表中的所有记录,以及左表中符合连接条件的记录 。
MySQL中的交叉连接、内连接、外连接学生student表:
课程course表:
交叉连接:
结果为两张表的笛卡尔积:
内连接(inner join,inner 可以省略)
显示内连接和隐示内连接获得的查询结果是一样的,都是A表和B表的交集(例:A.id = B.id),但是只能查到有关系的信息,如果A表的一条数据的与B表关联的字段没有对应的信息(如:A.id = null) , 即:这条数据在B表中没有对应的信息,则无法获得 。
外连接(outer join,outer可以省略)
左外连接获得的查询结果是左边的表A的全部信息和A,B两张表的交集 , 左边A表的全部包含A表中在B表中没有对应关系的信息 。
右外连接获得的查询结果是右边的表B的全部信息和A,B两张表的交集,右边B表的全部包含B表中在A表中没有对应关系的信息 。
mysql连接方式左连接mysql怎么表示外连接:返回包括左表中的所有记录和右表中联结字段相等的记录;
右连接:返回包括右表中的所有记录和左表中联结字段相等的记录;
全连接:返回两个连接表的所有记录;mysql没有全连接
内连接:返回两个连接表连接字段相同的记录;
union:对两个结果集进行并集mysql怎么表示外连接,并去重,当交换两个select语句顺序后,最终结果与不交换时无区别;
union all:对两个结果集进行并集,不去重,当交换两个select语句顺序后,最终结果与不交换时有区别;
举个例子 有一个学生表 班级 姓名 一个成绩表 姓名 成绩 mysql怎么表示外连接我们需要返回X班的所有学生的成绩,但是班上有人缺考,也就是成绩表里没有姓名,我们先用on得到的就是有考试成绩的名字,通过外连接,我们就可以得到全班人的名字以及成绩 。
关于mysql怎么表示外连接和mysql外连接和内连接的介绍到此就结束了,不知道你从中找到你需要的信息了吗 ?如果你还想了解更多这方面的信息,记得收藏关注本站 。
推荐阅读
- 动作角色类pc游戏,动作角色类pc游戏推荐
- 美图电商如何投诉店铺,美图投诉服务商电话
- asp.netexcel合并单元格值,vbnet excel合并单元格
- linux用命令查网卡 linux查看网卡信息 命令
- 包含点击大分类跳转到相应位置js的词条
- 战斗类即时战略游戏手游,战斗类即时战略游戏手游有哪些
- 区块链审计,区块链审计公司排行
- go语言声明一个列表 golang列表
- 神魔下载,神魔下载最新版