MySQL实战5 子查询

MySQL实战 目录

子查询介绍:出现在其他语句中的select语句,被包裹的select语句就是子查询或查询
包裹子查询的外部的查询语句:称主查询语句

案例:查询在位置编号为1700的部门里的所有员工的名字
SELECT last_name FROM employees WHERE department_id 
                IN(SELECT department_id FROM departments 
                                 WHERE location_id=1700
                                );

1.子查询分类

通过位置来分:
SELECT 后面:仅仅支持量子查询
FROM 后面:支持表查询
WHERE 或having 后面: 支持标量子查询(重要) 列子查询(重要) 行子查询(用的较少)
EXISTS 后面(相关查询):支持表子查询

按结果集的行列数不同的分类
标量子查询(结果集只有一行一列)
列子查询(结果集只有一列但有多行)
行子查询(结果集只有一行当有多列)
表子查询(结果集多行多列)

2.子查询特点

子查询放在小括号内
子查询一般放在条件但右侧
标量子查询,一般搭配着单行操作符来使用(> < >= <= <> =)
列子查询,一般搭配着多行操作符使用:in any/some all
子查询但执行顺序优先于主查询(SELECT后的子查询例外)

3.where后面的标量子查询

案例:查询工资比Abel这个人的高的员工信息
 SELECT * FROM employees
          WHERE  salary >(
                      SELECT salary
                              FROM employees
                                    WHERE last_name = 'Abel'
                    );
案例:查询job_id与141号员工相同,salary比143号员工多的员工姓名,job_id 和工资
SELECT last_name,job_id,salary FROM employees
                 WHERE job_id = (SELECT job_id FROM employees WHERE employee_id =141) 
                 AND salary > (SELECT salary FROM employees WHERE employee_id =143);
这个案例说明一个主查询里可以放很多个子查询
案例:子查询里用到分组查询函数:查询员工工资最少的员工的last_name,job_id和salary
 SELECT last_name,job_id,salary FROM employees
        WHERE salary = (SELECT min(salary) FROM employees  );
案例:查询最低工资大于50号部门最低工资的部门id和其最低工资
SELECT department_id,MIN(salary)
FROM employees 
GROUP BY department_id
HAVING MIN(salary)>(SELECT MIN(salary) FROM employees WHERE department_id =50   )

4.where后面的列子查询(行多子查询)

首先来看一下多行操作符:
in/not in:等于列表中的任意一个
a in(10,20,30); 可以替换 a=10 or a=20 or a=30
any/some:和子查询返回的某一个比较
a > any(10,20,30); 可以替换 a > min(10,20,30)
all:和子查询返回的所有值比较
a>all(10,20,30); 可以替换 a > max(10,20,30)

案例:返回location_id 是1400或1700的部门中的所有员工的名字
SELECT last_name FROM employees WHERE department_id
IN(SELECT department_id FROM departments WHERE location_id =1400 OR location_id=1700
);
案例:查询其他工种中比job_id为'IT_PROG'的员工某一工资低的员工的员工号,姓名,job_id和salary
   SELECT employee_id ,last_name,job_id,salary
          FROM employees
                WHERE 
                job_id <>'IT_PROG' AND
                
                salary  <   any(SELECT salary FROM employees WHERE job_id ='IT_PROG');
案例:查询其他工种中比job_id为'IT_PROG'的员工所有工资低的员工的员工号,姓名,job_id和salary
   SELECT employee_id ,last_name,job_id,salary
          FROM employees
                WHERE 
                job_id <>'IT_PROG' AND
                salary  <   ALL(SELECT salary FROM employees WHERE job_id ='IT_PROG');

5.where后面的执行子查询(一行多列)

案例:查询员工编号最小且工资最高的员工信息
SELECT *FROM employees 
        WHERE 
                   employee_id = (SELECT MIN(employee_id) FROM employees)
                       and 
                     salary =(SELECT MAX(salary) FROM employees);

执行子查询

 SELECT *FROM employees 
        WHERE (employee_id,salary)=(SELECT min(employee_id),max(salary)FROM employees);

6.SELECT 和FROM 后面的子查询

SELECT 后面(很少用到的,可以用前面讲的方法实现):
案例:查询每个部门的部门信息和对应的员工个数(不用连接查询)
    SELECT d.*,COUNT(e.employee_id) FROM departments d
             LEFT JOIN employees e
                     ON d.department_id = e.department_id
                     GROUP BY d.department_id;
                     
    SELECT d.*,( SELECT COUNT(*) FROM employees e WHERE d.department_id =e.department_id  )FROM departments d;
案例:查询员工号等于102的部门名(不用连接查询)
    SELECT department_name FROM departments WHERE department_id =(SELECT e.department_id FROM employees e WHERE e.employee_id=102);
    SELECT  (SELECT d.department_name FROM departments d WHERE d.department_id =  e.department_id ) FROM employees e WHERE e.employee_id=102;
案例:查询每个部门的平均工资等级
SELECT  AVG(e.salary) FROM employees e  GROUP BY e.department_id
                     
(SELECT  AVG(e.salary)  avg_sal FROM employees e  GROUP BY e.department_id) avg_sal_res 当成一个新的表
 

SELECT avg_sal_res.avg_sal,g.grade_level
            FROM
            (SELECT  AVG(e.salary)  avg_sal FROM employees e  GROUP BY e.department_id) avg_sal_res
            INNER JOIN job_grades g
            on avg_sal BETWEEN g.lowest_sal AND g.highest_sal;
                     

6.EXISTS后面(相关子查询)

exists的作用是:判断子查询有没有结果的存在

案例:
 SELECT EXISTS(SELECT employee_id FROM employees);
 
 SELECT EXISTS(SELECT employee_id FROM employees WHERE employee_id=99999);
案例:查询有员工的部门名,EXISTS 很少使用
  SELECT EXISTS(SELECT employee_id FROM employees WHERE employee_id=99999);
    SELECT department_name FROM departments d WHERE  EXISTS(SELECT * FROM employees e WHERE e.department_id = d.department_id);
    SELECT department_name FROM departments d WHERE department_id IN(SELECT e.department_id FROM employees e WHERE e.department_id = d.department_id);
注意

本文用到的myemployees.sql 文件在MySQL实战2 语法、筛选条件和函数 附件中

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,015评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,262评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,727评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,986评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,363评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,610评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,871评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,582评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,297评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,551评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,053评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,385评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,035评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,079评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,841评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,648评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,550评论 2 270