【MySQL】多表查询的使用

1. 熟悉常见的表

DESC employees;
DESC departments;
DESC locations;
引入:查询一个员工(ABEL)在哪个城市工作
SELECT *
FROM employees
WHERE last_name = 'Abel';

SELECT * 
FROM departments
WHERE department_id = 80;

SELECT *
FROM locations
WHERE location_id = 2500;

但是这样进行了三次查询,会比较浪费网络资源

2. 正确的多表查询

错误的实现方式
SELECT employeea_id,department_name
FROM employees,departments;

这样查询出了2889条记录
这样就像是每个员工在所有部门都干过,说明每个员工斗鱼每个部门匹配了一遍
这叫作:出现笛卡尔积的错误
笛卡尔积:又叫交叉连接
假设有两个集合x,Y,那么x和y的笛卡尔积就是x和y的所有可能组合

正确的实现方式:需要有连接条件
SELECT employee_id,department_name
FROM employees,departments
WHERE employees.`department_id` = departments.`department_id`;
# 着重号可以不写:``
延申:如果查询语句中出现了多个表中都存在的字段,则必须指明此字段所在的表
SELECT employee_id,department_name,departments.department_id
FROM employees,departments
WHERE employees.`department_id` = departments.`department_id`;

建议:都指明字段所在的表

3.可以给表起别名,在select和where中使用

SELECT emp.employee_id,emp.department_name,dept.department_id
FROM employees emp,departments dept
WHERE emp.department_id = dept.department_id;

但是:写了别名以后,一旦在select或者where中使用了别名的话,就必须使用别名!

练习:如果有n个表查询,则至少需要n-1个连接条件
SELECT employee_id,department_name,d.department_id,city
FROM employees e,departments d,locations l
WHERE e.department_id = d.department_id
AND d.location_id = l.location_id;

4.多表查询的分类

  • 角度一:
    • 等值连接和非等值连接
  • 角度二:
    • 自连接和非自连接
  • 角度三:
    • 内连接和外连接

等值连接:就是上面写的几条,有等好的语句
非等值连接:

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

自连接和非自连接:

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

内连接和外连接

内连接:合并具有同一列的两个以上的表的行,结果集中不包含一个表与另一个表不匹配的行(即交集)
SELECT employee_id,department_name
FROM employees e,departments d
WHERE e.`department_id` = d.department_id;
# 只有106条记录

外连接:合并具有同一列的两个以上的表的行,结果集中除了包含一个表与另一个表不匹配的行之外,还查询到了左表或右表中不匹配的行

外连接又分为:(类似于韦恩图)
请添加图片描述

  • 左外连接:59+41
    • 左外连接:合并具有同一列的两个以上的表的行,结果集中除了包含一个表与另一个表不匹配的行之外,还查询到了左表中不匹配的行
  • 右外连接:39+41
    • #####右外连接:合并具有同一列的两个以上的表的行,结果集中除了包含一个表与另一个表不匹配的行之外,还查询到了右表中不匹配的行
  • 满外连接:59+41+39
    • 满连接:合并具有同一列的两个以上的表的行,结果集中除了包含一个表与另一个表不匹配的行之外,还查询到了左表和右表中不匹配的行

请添加图片描述


5.MYSQL中内外连接的实现:

问题:查询所有的员工的last_name,department_name信息.

只要是查询所有的信息,就要考虑外连接!!!

SQL92语法(可惜的是,MySQL不支持,我们了解就行)

SQL92语法的内连接实现:
SELECT employee_id,department_name
FROM employees e,departments d
WHERE e.`department_id` = d.department_id;
SQL92语法的外连接实现:使用 +

此题条件:左表中有个人没有部分数据,但是我们必须要把他显示出来,需要使用左外

SELECT employee_id,department_name
FROM employees e,departments d
WHERE e.`department_id` = d.department_id(+);

记忆小技巧:想象人的两条腿,哪条腿短,就将哪条腿垫高(+)
这样在mysql中是错误的

SQl99语法(mysql支持这种方式:)

使用JOIN…ON的方式实现多表的查询,这种方式也能解决外连接的问题

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`;
SQL99实现左外连接(outer可以省略):加上了那些没有部门的人
SELECT last_name,department_name
FROM employees e LEFT OUTER 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`;
满外连接:

但是:MySQL不支持FULL OUTER JOIN

SELECT last_name,department_name
FROM employees e FULL OUTER JOIN departments d
ON e.`department_id` = d.`department_id`;

解决办法:union和union all语句


6.union和union all语句

UNION : 自动去掉中间的一份重复部分(还剩一份)
UNION ALL :不去掉中间的重复部分(两份交集)
能用union ALL 就不要用union(效率低了)
请添加图片描述

7种jion的实现:
  1. 中图:内连接:
SELECT last_name,department_name
FROM employees e JOIN departments d
ON e.`department_id` = d.`department_id`;
  1. 左上图:左外
SELECT last_name,department_name
FROM employees e LEFT OUTER JOIN departments d
ON e.`department_id` = d.`department_id`;
  1. 右上图:右外
SELECT last_name,department_name
FROM employees e RIGHT OUTER JOIN departments d
ON e.`department_id` = d.`department_id`;
  1. 左中图:
SELECT last_name,department_name
FROM employees e LEFT OUTER JOIN departments d
ON e.`department_id` = d.`department_id`
WHERE d.department_id IS NULL;
  1. 右中图
SELECT last_name,department_name
FROM employees e RIGHT OUTER JOIN departments d
ON e.`department_id` = d.`department_id`
WHERE e.department_id IS NULL;

6.左下图(满外)

  • 方法一.左上图 UNION ALL 右中图
SELECT last_name,department_name
FROM employees e LEFT OUTER JOIN departments d
ON e.`department_id` = d.`department_id`
UNION ALL
SELECT last_name,department_name
FROM employees e RIGHT OUTER JOIN departments d
ON e.`department_id` = d.`department_id`
WHERE e.department_id IS NULL;
  • 方法二.左中图 UNION ALL 右上图
SELECT last_name,department_name
FROM employees e LEFT OUTER JOIN departments d
ON e.`department_id` = d.`department_id`
WHERE d.department_id IS NULL
UNION ALL
SELECT last_name,department_name
FROM employees e RIGHT OUTER JOIN departments d
ON e.`department_id` = d.`department_id`;

7.右下图:左中图 UNION ALL 右中图

SELECT last_name,department_name
FROM employees e LEFT OUTER JOIN departments d
ON e.`department_id` = d.`department_id`
WHERE d.department_id IS NULL
UNION ALL
SELECT last_name,department_name
FROM employees e RIGHT OUTER JOIN departments d
ON e.`department_id` = d.`department_id`
WHERE e.department_id IS NULL;

7. SQL99语法的新特性1:自然连接

可以把自然连接理解为SQL92中的等值连接,他会帮你自动查询两张连接表中的所有相同的字段,然后进行等值连接。

虽然很简洁,但是并不灵活

SELECT employee_id,last_name,department_name
FROM employees e NATURAL JOIN departments d;


8. SQL99语法的新特性2:using的使用

但是这个不适用于自连接!!!

SELECT employee_id,last_name,department_name
FROM employees e JOIN departments d
USING(department_id);
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值