SQL|多表查询

多表查询,也称为关联查询,指两个或者更多个表一起完成查询操作。

前提条件:这些一起查询的表之间是有关系的(一对一、一对多),他们之间一定是有关联字段,这个关联字段可能建立了外键,也有可能没有建立外键。比如:员工表和部门表,这两个表依靠“部门编号”进行关联。

笛卡尔积(或交叉连接)的理解

笛卡尔乘积是一个数学运算。假设我有两个集合x和y,那么x和y的笛卡尔积就是x和y的所有可能组合,也就是第一个对象来自于x,第二个对象来自于y的所有可能。组合的个数即为两个集合中元素个数的乘积数。

 SQL92中,笛卡尔积也称为交叉连接,英文是CROSS JOIN。在SQL99中也是使用CROSS JOIN表示交叉连接。它的作用就是可以把容易表进行连接,即使这两张表不相关。在MySQL中如下情况会出现笛卡尔积:

#查询员工姓名和所在部门名称
SELECT last_name, department_name FROM employees,departments;
SELECT last_name, department_name FROM employees CROSS JOIN departments;
SELECT last_name, department_name FROM employees INNER JOIN departments;
SELECT last_name, department_name FROM employees JOIN departments;
  • 笛卡尔积的错误会在下面条件下产生
    • 省略多表的连接条件(或关联条件)
    • 连接条件(或关联条件)无效
    • 所有表中的所有行互相连接
  • 为了避免笛卡尔积,可以在WHERE加入有效的连接条件
  • 加入连接条件后,查询语法: 
SELECT table1.column, table2.column
FROM table1, table2
WHERE table1.column1 = table2.column2; #连接条件
  •  在WHERE子句中写入连接条件。
  • 在表中有相同列时,在列名之前加上表名前缀。

如果查询语句中出现了多个表中都存在的字段,则必须指名此字段所在的表。

建议:从sql优化的角度,建议多表查询时,每个字段都指明其所在的表

多表查询分类

角度1:等值连接vs非等值连接(例子)

#非等值连接
SELECT * 
FROM job_grades;

SELECT last_name,salary,grade_level
FROM employees e,job_grades j
WHERE e.`salary` BETWEEN j.`lowest_sal` AND j.`highest_sal`;

拓展1:多个连接条件与AND操作符

拓展2:区分重复的列名

  • 多个表中有相同的列时,必须在列名之间加上表名前缀;
  • 在不同表中具有相同列名的列可以用表名加以区分 。
SELECT last_name, department_name 
FROM employees,departments
WHERE employees.`department_id`=departments.`department_id`;

拓展3:表的别名

  • 使用别名可以简化查询
  • 列名前使用表名前缀可以提高查询效率 

#可以给表起别名,在select和where中使用表的别名
#如果给表起了别名,一旦在select或where中使用表名的话,则必须使用表的别名,而不能使用表的原名
SELECT emp.`employee_id`, dept.`department_name`,emp.`department_id` 
FROM employees emp,departments dept
WHERE emp.`department_id`=dept.`department_id`;

 拓展4:连接多个表

总结:连接n个表,至少需要n-1个连接条件。比如,连接三个表,至少需要两个连接条件。

#练习:查询员工的employee_id,last_name,department_name,city
SELECT employee_id,last_name,department_name,city
FROM employees e,departments d,locations l
WHERE e.`department_id`=d.`department_id`
AND d.`location_id`=l.`location_id`;

角度2:自连接(例子)vs非自连接

#练习:查询员工id,员工姓名,及管理者的id和姓名
SELECT emp.employee_id,emp.last_name,mgr.employee_id,mgr.last_name
FROM employees emp, employees mgr
WHERE emp.`employee_id`=mgr.`employee_id`;

角度3:内连接vs外连接 

内连接: 合并具有同一列的两个以上的表的行,结果集中不包含一个表与另外一个表不匹配的行;

外连接:两个表在连接过程中除了返回满足连接条件的行以外返回左(或者右)表中不满足条件的行,这种连接称为左(或右)外连接。没有匹配的行时,结果表中相应的值为空(NULL)。

  • 如果是左外连接,则连接条件中左边的表也称为主表,右边的表也称为从表。
  • 如果是右外连接,则连接条件中右边的表也称为主表,左边的表也称为从表。

SQL99语法实现多表查询

内连接

使用JOIN...ON子句创建连接的语法结构:

SELECT table1.column, table2.column, table3.column
FROM table1 
    JOIN table2 ON table1和table2的连接条件
        JOIN table3 ON table2和table3的连接条件

 语法说明:

  • 可以使用ON子句指定额外的连接条件;
  • 这个连接条件是与其他条件分开的;
  • ON子句是语句具有更高的易读性;
  • 关键字JOIN、INNER JOIN、CROSS JOIN的含义是一样的,都表示内连接
#SQL99实现内连接
SELECT last_name,department_name,city
FROM employees e JOIN departments d
ON e.`department_id`= d.`department_id`
JOIN locations l
ON d.location_id = l.location_id;

 外连接

 左外连接

SELECT 字段列表
FROM A表 LEFT JOIN B表
ON 关联条件
WHERE 等其他子句;
##SQL99实现外连接
#左外连接
SELECT last_name,department_name
FROM employees e LEFT JOIN departments d
ON e.`department_id`= d.department_id;

 右外连接

#右外连接
SELECT last_name,department_name
FROM employees e RIGHT OUTER JOIN departments d
ON e.`department_id`= d.department_id;

满外连接

  • 满外连接的结果=左右表匹配的数据+左表没有匹配到的数据+右表没有匹配到的数据;
  • SQL99是支持满外连接的。使用FULL JOIN或FULL OUTER JOIN来实现
  • 需要注意的是,MySQL不支持FULL JOIN,但是可以用LEFT JOIN UNION RIGHT JOIN代替。 

UNION的使用

合并查询结果 利用UNION关键字,可以给出多条SELECT语句,并将他们的结果组合成的单个结果集。合并时,两个表对应的列数和数据类型必须相同,并且相互对应。各个SELECT语句之间使用UNION或UNION ALL关键字分隔。

语法格式:

SELECT column,...FROM table1
UNION [ALL]
SELECT column,...FROM table2

 UNION操作符:返回两个查询的结果集的并集,去除重复记录;

UNION ALL操作符:返回两个查询的结果集的并集。对于两个结果集的重复部分不去重。

注意:执行UNION ALL语句时所需要的资源比UNION语句少。如果明确知道合并数据后的结果数据不存在重复数,或者不需要去除重复的数据,则尽量使用UNION ALL语句,以提高数据查询的效率。

七种sql JOINS的实现

#UNION的使用
#中图:内连接
SELECT employee_id, department_name 
FROM employees e JOIN departments d
ON e.`department_id`=d.`department_id`; 

#左上图:左外连接
SELECT employee_id, department_name 
FROM employees e LEFT JOIN departments d
ON e.`department_id`=d.`department_id`; 

#右上图:右外连接
SELECT employee_id, department_name 
FROM employees e RIGHT JOIN departments d
ON e.`department_id`=d.`department_id`; 

#左中图
SELECT employee_id, department_name 
FROM employees e LEFT JOIN departments d
ON e.`department_id`=d.`department_id`
WHERE d.`department_id` IS NULL;

#右中图
SELECT employee_id, department_name 
FROM employees e RIGHT JOIN departments d
ON e.`department_id`=d.`department_id`
WHERE e.`department_id` IS NULL;

#左下图
#方式1:左上图 UNION ALL 右中图
SELECT employee_id, department_name 
FROM employees e LEFT JOIN departments d
ON e.`department_id`=d.`department_id`
UNION ALL
SELECT employee_id, department_name 
FROM employees e RIGHT JOIN departments d
ON e.`department_id`=d.`department_id`
WHERE e.`department_id` IS NULL;

#方式2:左中图 UNION ALL 右上图
SELECT employee_id, department_name 
FROM employees e LEFT JOIN departments d
ON e.`department_id`=d.`department_id`
WHERE d.`department_id` IS NULL
UNION ALL
SELECT employee_id, department_name 
FROM employees e RIGHT JOIN departments d
ON e.`department_id`=d.`department_id`; 

#右下图 左中图 UNION ALL 右中图
SELECT employee_id, department_name 
FROM employees e LEFT JOIN departments d
ON e.`department_id`=d.`department_id`
WHERE d.`department_id` IS NULL
UNION ALL
SELECT employee_id, department_name 
FROM employees e RIGHT JOIN departments d
ON e.`department_id`=d.`department_id`
WHERE e.`department_id` IS NULL;

 

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值