(六)MySQL多表
多表查询即连接查询,使用一个select 语句查询多张表,即多表查询。
内连接(JOIN / INNER JOIN)
- 从一张表中取出所有的记录去另外一张表中匹配:利用匹配条件进行匹配,成功了则保留,失败了放弃。
- 如果内连接没有条件(允许),那么其实就是交叉连接(避免)。
- 内连接因为不强制必须使用匹配条件(on)因此可以在数据匹配完成之后,使用where条件来限制,效果与on一样(建议使用on)。
- 内连接通常是在对数据有精确要求的地方使用:必须保证两种表中都能进行数据匹配。
SELECT <字段名> FROM <表1> <别名> INNER JOIN <表2> [ON子句] INNER JOIN <表3> [ON子句]
自然连接(NATURAL JOIN)
- 连接时不需要额外指定连接条件,在等值连接中把目标中重复的属性列去掉的连接查询。
自连接
- 表与它自己进行的连接查询,必须设置表别名
-
假设在【成绩表】有【主键ID】【学生姓名】【课程名称】【成绩】等字段。
现在要查询 “语文成绩>=数学成绩”的学生姓名,这时就可以使用自连接查询:
select 【学生姓名】
from 【成绩表】 AS a,【成绩表】 AS b
where a.【主键ID】=b.【主键ID】
and a.【成绩】>=b.【成绩】
and a.【课程名称】='语文'
and b.【课程名称】='数学'
外连接(left join / left outer join, right join / right outer join, union)
- 按照某一张表作为主表(表中所有记录在最后都会保留),根据条件去连接另外一张表,从而得到目标数据。
-
1、 确定连接主表:左连接就是left join左边的表为主表;right join就是右边为主表
2、 拿主表的每一条记录,去匹配另外一张表(从表)的每一条记录
3、 如果满足匹配条件:保留;不满足即不保留
4、 如果主表记录在从表中一条都没有匹配成功,那么也要保留该记录:从表对应的字段值都未NULL
- 外连接中主表数据记录一定会保存:连接之后不会出现记录数少于主表(内连接可能)
- 左连接和右连接其实可以互相转换,但是数据对应的位置(表顺序)会改变
左连接
- 左表是主表;
SELECT <字段名> FROM <主表> LEFT OUTER JOIN <从表> <ON子句>
右连接
- 右表是主表;
SELECT <字段名> FROM <从表> RIGHT OUTER JOIN <主表> <ON子句>
交叉连接(CROSS JOIN)
- 将两张表的数据与另外一张表彼此交叉。
-
1、从第一张表依次取出每一条记录
2、 取出每一条记录之后,与另外一张表的全部记录挨个匹配
3、 没有任何匹配条件,即省略where 子句,所有的结果都会进行保留,返回笛卡尔积。
4、 记录数 = 第一张表记录数 * 第二张表记录数;字段数 = 第一张表字段数 + 第二张表字段数(笛卡尔积)
SELECT <字段名> FROM <表1> CROSS JOIN <表2> CROSS JOIN <表3> [WHERE子句] #或 SELECT <字段名> FROM <表1>, <表2>,<表3> [WHERE子句]
USING关键字
是在连接查询中用来代替对应的on关键字的,进行条件匹配。
原理
1、 在连接查询时,使用on的地方用using代替
2、 使用using的前提是对应的两张表连接的字段是同名(类似自然连接自动匹配)
3、 如果使用using关键字,那么对应的同名字段,最终在结果中只会保留一个。
语法
基本语法:表1 [inner,left,right] join 表2 using(同名字段列表); //连接字段
总结:应避免使用交叉连接,看情况使用内连接或者外连接,当要以一表数据为主,使得查询的记录数不会少于此表记录数时考虑外连接,当严格按照条件进行匹配时,使用内连接比较合适,当两表条件字段名一致时,可以使用using关键字。
子查询
- 子查询指将一个查询语句嵌套在另一个查询语句中。子查询可以在WHERE、 SELECT、UPDATE 和 DELETE 语句中使用,而且可以进行多层嵌套。
- 操作符可以是 比较运算符(=,>,>=,<,<=,<>) 和 IN、NOT IN、ANY、EXISTS、NOT EXISTS 等关键字。
- 一般来说,表连接(内连接和外连接等)都可以用子查询替换,但反过来却不一定。子查询适合作为查询的筛选条件,而表连接更适合于查看连接表的数据。
- 子查询里返回值若有重复值则会被抛弃重复值。
在WHERE 中的语法格式:
WHERE <表达式> <操作符> (子查询)
在SELECT中:
SELECT (子查询) FROM 表名;
嵌套在 SELECT 语句的 FROM 子句中:
SELECT * FROM (子查询) AS 表的别名;