MySQL高级查询与编程笔记 • 【第3章 子查询】
全部章节 >>>>
本章目录
3.1 子查询定义和单行子查询
3.1.1 子查询定义
子查询(subquery)是数据库经常用到的一个操作,它不仅用在数据查询语句中,在 DML 语句中也都会用到子查询
子查询将一个查询语句嵌套(nest)在另一个查询语句中,在特定情况下,一个查询语句的条件需要另一个查询语句来获取,内层查询语句的查询结果可以为外层查询语句提供查询条件
子查询的实质:一个 select 语句的查询结果能够作为另一个语句的输入值。子查询不仅可用于 where 子句中,还能够用于 from 子句中,此时子查询的结果将作为一个临时表(temporary table)来使用
子查询还能以字段的形式出现在 select 语句的选择列中。根据子查询所返回的结果行数,可以将其分为单行子查询和多行子查询
3.1.2 单行子查询应用
单行子查询指子查询的返回结果只有一行数据。当在主查询的条件语句中引用子查询的结果时,可使用单行比较符(如=、>、<、>=、<= 和 < >)进行比较
查询“战争”类题材电影的具体信息,要求输出片名和导演名
示例:
(1)电影(movie)表包含电影名、导演名和电影类型编号信息,但并不包含电影类型名称信息;电影类型(movie_type)表既包含电影类型编号信息又包含电影类型名称信息。上述这两张表的共同信息是电影类型编号,所以查询时需连接电影表和电影类型表,并以电影类型编号作为两表的连接。首先,从电影类型表查询出类型名称为“战争”的电影类型编号,使用 SQL1 作为标记
select id from movie_type where typeName=' 战争 '
(2)根据 SQL1 查询出的地区编号,在电影表中检索出电影名和导演名信息。使用 SQL2 作为标记,并将 SQL1作为查询条件代入 SQL2
select movieName 电影名 ,director 导演名 from movie where typeId=
(select id from movie_type where typeName=' 战争 ')
由于在 movie_type 表中 typeName 是唯一约束列,因而子查询 SQL1 的执行结果只能有 1 条(单行子查询)或 0 条记录。
此例还可以采用连接查询实现
select movieName 电影名 ,director 导演名 from movie m, movie_type mt
where m.typeId=mt.movie_type and typeName=' 战争 '
示例:
查询票价高于平均票价的电影信息,要求输出电影名和导演名
(1)获得平均票价,使用 SQL1 标记
select avg(ticketPrice) from movie
(2)查询票价大于平均票价的电影信息,要求输出电影名和导演名。使用 SQL2 标记,并将 SQL1 作为查询条件代入 SQL2
select movieName 电影名 ,director 导演名 from movie where ticketPrice>( select avg(ticketPrice) from movie)
示例:
查询票价高于“战争”类题材的电影信息,要求输出电影名和导演名
(1)查询类型名为“战争”的电影类型编号,使用 SQL1 标记
select id from movie_type where typeName=' 战争 '
(2)查询“战争”类题材电影的平均票价,使用 SQL2 标记,执行时将 SQL1 作为查询条件代入 SQL2
select avg(ticketPrice) from movie where typeID=(select id from movie_type where typeName=' 战争 ')
(3)查询票价大于“战争”类题材电影平均票价的电影信息,要求输出电影名和导演名。查询语句使用 SQL3 标记,使用时将 SQL2 作为条件代入 SQL3
select movieName 电影名 ,director 导演名 from movie where ticketPrice>
(select avg(ticketPrice) from movie where typeID=(
select id from movie_type where typeName=' 战争 '))
子查询应用经验初步归纳如下:
- 子查询一般用于 select 语句的 where 子句中,且可以嵌套
- 编写复杂的子查询的解决思路是逐层分解查询,即从最内层的子查询开始分解,将嵌套的 SQL 语句拆分为一个个独立的 SQL 语句
- 子查询的执行过程遵循“由里及外”的原则,即先执行最内层的子查询语句,然后将执行结果与外层的语句进行合并,依次逐层向外扩展并最终形成完整的 SQL 语句
- 一般情况下,连接查询可改为子查询实现;但子查询却不一定可改为连接查询实现
- 子查询与连接查询执行效率的比较:当子查询执行结果的行数较大,而主查询执行结果的行数较小时,子查询执行效率较高;反之,则连接查询执行效率较高
3.1.4 实践练习
3.2 多行子查询应用
3.2.1 in 比较符
使用多行比较符 in 时,主查询会与子查询中的每一个值进行比较,如果与其中的任意一个值相同,则返回。not in 与 in 的含义恰好相反
查询“战争”和“喜剧”类题材电影的相关信息,要求输出片名和导演名
示例:
(1)查询类型名为“战争”和“喜剧”的电影类型编号
select id from movie_type where typeName=' 战争 ' or typeName=' 喜剧 '
(2)查询“战争”和“喜剧”类题材电影的相关信息
select movieName 电影名 ,director 导演名 from movie where typeID in
(select id from movie_type where typeName=' 战争 ' or typeName=' 喜剧 ')
由于多行子查询返回的结果行数可以为一个,因而单行子查询也是多行子查询的一种特殊情况,所以单行子查询的“=”比较符可以替换为多行子查询的“in”比较符。但不能将多行子查询的“in”比较符替换为单行子查询的“=”比较符。
查询客户“zhang01”顾客所预订电影的具体信息,要求输出电影名、导演名、票价和片长,并按照票价升序排列
示例:
(1)查询客户“zhang01”的客户编号
select id from customer where username='zhang01'
(2)查询客户“zhang01”所预订的所有电影的排片编号
select scheduleId from ticket_sell where customerID in
(select id from customer where username='zhang01')
(3)查询客户“zhang01”所预订的所有电影的电影编号
select movieId from `schedule` where id in (select scheduleId from ticket_sell where customerID in (select id from customer where username='zhang01'))
(4)查询客户“zhang01”所预订电影的具体信息
select movieName 电影名 ,director 导演名 ,ticketPrice 票价(元),filmLength 片长(分钟)
from movie where id in(select movieId from `schedule` where id in
(select scheduleId from ticket_sell where customerID in
(select id from customer where username='zhang01'))) order by ticketPrice
此示例嵌套较深,如果采用连接查询写法会相对简洁,特别是当商品记录数较大的情况下,连接查询的效率会更高。使用连接查询实现的 SQL 语句如下:
select distinct movieName 电影名 ,director 导演名 ,ticketPrice 票价(元),filmLength 片长(分钟) from movie m,`schedule` s,ticket_sell ts,customer c where m.id=s.movieId
and s.id=ts.scheduleId and ts.customerId=c.id and c.username='zhang01' order by ticketPrice
上面例子的子查询实现很好地体现了子查询应用的解决之道──“由里及外”的原则,即逐层生成包含子查询在内的主查询,逐层生成的主查询又成为上一层主查询所包含的子查询,如此依次递进(recursion),最终生成最上层的包含所有子查询的主查询。
all 关键字位于多行比较运算符之后,通过 all 关键字将一个表达式或列的值与子查询所返回的一列值中的每一行进行比较,只要有一次比较的结果为 false(假),则 all 测试返回 false,主查询不执行;否则返回 true,执行主查询
all 运算符的含义如下:
- 当 <all 时,表示小于最小值
- 当 >all 时,表示大于最大值
语法:
表达式或字段 多行比较运算符 all( 子查询 )
查询比所有“喜剧”类题材电影的票价都高的电影信息,要求输出片名和导演名
示例:
(1)查询类型为“喜剧”的电影类型编号
select id from movie_type where typeName=' 喜剧 '
(2)查询所有“喜剧”类题材电影的票价
select ticketPrice from movie where typeID=(
select id from movie_type where typeName=' 喜剧 ')
(4)查询比所有“战争”类题材电影的票价都高的电影信息
select movieName 电影名 ,director 导演名 from movie where ticketPrice > all (select ticketPrice from movie where typeID=(select id from movie_type where typeName=' 喜剧 '))
由于“>all(子查询)”的含义是“大于子查询返回结果的最大值”,所以还可以采用“>(子查询所获取的最大列值)”的方式求解上面这个例子,该方法的 SQL 语句如下:
select movieName 电影名 ,director 导演名 from movie where ticketPrice > (select max(ticketPrice) from movie where typeID=( select id from movie_type where typeName=' 喜剧 ' ))
3.2.3 any|some 关键字子查询
any 与 some 的查询功能相同
any 或 some 用于子查询之前,通过 any|some 比较运算符,将一个表达式或列的值与子查询所返回的一列值中的每一行进行比较,只要有一次比较的结果为 true,则 any 或 some 测试返回 true,主查询执行;否则结果为false,主查询不执行
any|some 运算符的含义如下:
- 当 <any|some 时,表示小于最大值
- 当 =any|some 时,表示与 in 运算符等价
- 当 >any|some 时,表示大于最小值
语法:
表达式或字段 多行比较运算符 any|some(子查询)
查询比任意一个“喜剧”类题材电影的票价高的电影信息,要求输出电影名和导演名
示例:
select movieName 电影名 ,director 导演名 from movie where ticketPrice > any
(select ticketPrice from movie where typeID=(
select id from movie_type where typeName=' 喜剧 '))
由于“>any(子查询)”的含义是“大于子查询返回结果的最小值”,所以还可以采用“>(子查询所获取的最小列值)”的方式求解上面这个例子,该方法的 SQL 语句如下:
select movieName 电影名 ,director 导演名 from movie where ticketPrice > (select min(ticketPrice) from movie where typeID=( select id from movie_type where typeName=' 喜剧 ' ))
3.2.4 实践练习
3.3 子查询特殊应用
3.3.1 from 子句中的子查询
子查询通常用于 where 子句中,但其也可在 from 子句和 select 子句中使用
示例:
影院在线售票系统为了提升影片的上座率,优化票价定价的科学性。为了了解每一个影片的票价与该类题材电影整体票价的对照关系,比较合适的做法是在显示每一个影片票价的同时,显示所属类型电影的平均票价
select mt.typeName 电影类型 , m.movieName 电影名 , m.director 导演名 , round(m.ticketPrice,2)
票价(元), round(A.avgPrice,2) 该电影类型平均票价(元) from movie m, movie_type mt,
(select typeId,avg(ticketPrice) avgPrice from movie group by typeId) A where m.typeId=mt.id and m. typeId=A.typeId order by mt.id
3.3.2 select 子句中的子查询
在 select 子句中使用子查询,其实质是将子查询的执行结果作为 select 子句的列,可以起到与连接查询异曲同工的作用
示例:
分别获取张艺谋所导演影片的上映数量以及上映班次
(1)获取张艺谋所导演影片的上映数量
select count(*) 张艺谋所导演影片的上映数量 from movie where director=' 张艺谋 '
(2)获取张艺谋所导演影片的上映班次
select count(movieId) 张艺谋所导演影片的上映班次 from `schedule` where movieId in
(select id from movie where director=' 张艺谋 ')
(3)将(1)和(2)获得的查询结果,即张艺谋所导演影片的上映数量和上映班次作为 select 子句的查询列,其形式即为 select 子句中的子查询。
select count(*) 张艺谋所导演影片的上映数量 , (select count(movieId) from `schedule` where
movieId in (select id from movie where director=' 张艺谋 ')) 张艺谋所导演影片的上映班次
from movie where director=' 张艺谋 '
exists 用于检查子查询是否至少会返回一行数据,该子查询实际上并不返回任何数据,而是返回值 true 或false。exists 指定一个子查询,用于检测行的存在。当子查询的行存在时,则执行主查询表达式,否则不执行
查询所有通过影院在线售票系统预订电影票的客户姓名
语法:
主查询表达式 [not] exists ( 子查询 )
示例:
(1)主查询用于从客户表获取客户姓名。
(2)exists 指定的子查询将从售票表中获取满足“客户编号 = 客户表 . 客户编号”条件的任意数据。
(3)只要 exists 子查询的结果集中有数据行返回,exists 子查询的返回结果若为 true,则执行主查询获得所有预订电影票的客户姓名;exists 子查询的返回结果若为 false,则不执行主查询。最终的 SQL 语句如下:
select customerName 客户姓名 from customer c where exists
(select * from orders where customerID=c.customerID)
3.3.4 实践练习
3.4 DML 语句中的子查询
3.4.1 update 子句中的子查询
- 子查询不仅可在 select 语句中使用,以实现需要嵌套的查询功能,还可以维护数据,完成复杂的更新、删除和插入功能
- 为了完成上述数据维护功能,需要在 DML 的 update 语句、delete 语句和 insert 语句中使用子查询
- 在 DML 语句中使用子查询与在 select 语句中使用子查询的原理是一致的,均为将内层子查询的结果作为外层主查询中 where 条件的参考值来使用
示例:
为响应政府提升公民灾难意识和应对能力的号召,院线将所有灾难片电影的票价降低 20%
(1)在子查询中获取类型为“灾难”的电影类型编号。
(2)在主查询中,使用 update 语句将所有灾难片电影的票价降低 20%
update product set currentPrice=currentPrice*0.9 where categoryID in(
select categoryID from category where categoryName=' 灾难 ')
示例:
计算所有客户预订电影票的总金额,并使用该金额更新客户表中“累计订票金额”的字段值
(1)在售票表中,根据客户编号进行分组,并使用“sum( 实际票价 )”汇总出每个客户的总订票金额
select customerID 客户编号 , sum(purchasePrice) 总订票金额(元) from ticket_sellgroup by customerID
(2)因为要将(1)中汇总出的每个客户的总订票金额赋给客户表中的“累计订票金额”字段,所以可以将(1)中的 SQL 作为子查询,并在主查询中执行“update 客户表 set 累计购票金额 =( 子查询中获取的每个客户的总订票金额 )”。
(3)为实现 update 语句,需在子查询中删除选择列“客户编号”,并且为了建立主查询与子查询的关联,还需要在子查询的 where 条件中设定“客户编号 = 售票表 . 客户编号”
update customer c set totalFee=(select sum(purchasePrice) from ticket_sell where customerID=c.ID group by customerID)
使用子查询删除客户“chen01”
示例:
删除数据时需要考虑表的主从关系,正确的做法是先删除从表数据,再删除主表数据。
(1)使用子查询删除售票表中客户“chen01”所有的订票记录
delete from ticket_sell where customerID in
(select id from customer where username='chen01')
(2)删除客户表中客户“chen01”的记录
delete from customer where username='chen01'
3.4.3 实践练习
总结:
- 子查询将一个查询语句嵌套在另一个查询语句中,在特定情况下,一个查询语句的条件需要另一个查询语句来获取
- 比较运算符 all 关键字用于子查询之前,通过该关键字将一个表达式或列的值,与子查询所返回的一列值中的每一行进行比较
- exists 用于检测行的存在,该子查询实际上并不返回任何数据,而是返回值 true 或false。当子查询的行存在时,则执行主查询表达式,否则不执行
- 在 DML 语句中使用子查询与在 select 语句中使用子查询的原理是一致的,均为将内层子查询的结果作为外层主查询中 where 条件的参考值来使用
MySQL高级查询与编程笔记 • 【第3章 子查询】的更多相关文章
- MySQL高级查询与编程笔记 • 【目录】
章节 内容 实践练习 MySQL高级查询与编程作业目录(作业笔记) 第1章 MySQL高级查询与编程笔记 • [第1章 数据库设计原理与实战] 第2章 MySQL高级查询与编程笔记 • [第2章 数据 ...
- MySQL学习笔记(五)—— 子查询及联结
子查询: 子查询,即嵌套在其他查询中的查询.例如我们有这样几个表,顾客表,订单表,商品表,我们想知道有哪些客户买了商品A,那么我们就需要先查看哪些订单里包含了商品A,然后根据订单查出是哪些客户. my ...
- MySQL数据库学习笔记(六)----MySQL多表查询之外键、表连接、子查询、索引
本章主要内容: 一.外键 二.表连接 三.子查询 四.索引 一.外键: 1.什么是外键 2.外键语法 3.外键的条件 4.添加外键 5.删除外键 1.什么是外键: 主键:是唯一标识一条记录,不能有重复 ...
- MySQL数据库学习笔记----MySQL多表查询之外键、表连接、子查询、索引
本章主要内容: 一.外键 二.表连接 三.子查询 四.索引 一.外键: 1.什么是外键 2.外键语法 3.外键的条件 4.添加外键 5.删除外键 1.什么是外键: 主键:是唯一标识一条记录,不能有重复 ...
- mysql学习笔记-- 多表查询之外键、表连接、子查询、索引
本章主要内容: 一.外键 二.表连接 三.子查询 四.索引 一.外键: 1.什么是外键 2.外键语法 3.外键的条件 4.添加外键 5.删除外键 1.什么是外键: 主键:是唯一标识一条记录,不能有重复 ...
- MySQL多表查询之外键、表连接、子查询、索引
MySQL多表查询之外键.表连接.子查询.索引 一.外键: 1.什么是外键 2.外键语法 3.外键的条件 4.添加外键 5.删除外键 1.什么是外键: 主键:是唯一标识一条记录,不能有重复的,不允许为 ...
- 警惕 MySql 更新 sql 的 WHERE 从句中的 IN() 子查询时出现的性能陷阱
警惕 MySql 更新 sql 的 WHERE 从句中的 IN() 子查询时出现的性能陷阱 以下文章来源:https://blog.csdn.net/defonds/article/details/4 ...
- Mysql常用sql语句(19)- in / exists 子查询
测试必备的Mysql常用sql语句系列 https://www.cnblogs.com/poloyy/category/1683347.html 前言 子查询在我们查询方法中是比较常用的,通过子查询可 ...
- SQL笔记-第八章,子查询
一.SELECT列表中的标量子查询 查询每种书籍类型中的最早出版的书籍.在SQL 查询中,需要将一本书籍的出版年份与该类型的所有书籍的出版年份进行比较,并且仅仅在它们匹配时,才返回一个记录 SELEC ...
随机推荐
- 容器之分类与各种测试(四)——unordered-multiset
unordered-multiset是不定序关联式容器,其底部是通过哈希表实现功能. (ps:黑色框就是bucket,白色框即为bucket上挂载的元素) 为了提高查找效率,bucket(篮子)的数量 ...
- Android中的性能优化
由于手机硬件的限制,内存和CPU都无法像pc一样具有超大的内存,Android手机上,过多的使用内存,会容易导致oom,过多的使用CPU资源,会导致手机卡顿,甚至导致anr.我主要是从一下几部分进行优 ...
- oracle extract
select extract(year from systimestamp) year ,extract(month from systimestamp) month ...
- 转 Android应用开发必备的20条技能
https://blog.csdn.net/u012269126/article/details/52433237 有些andorid开发人员感觉很迷茫,接下来该去看系统源码还是继续做应用,但是感觉每 ...
- 【Linux】【Services】【SaaS】Spinnaker
1. 简介 1.1. 说明: Spinnaker 是 Netflix 的开源项目,是一个持续交付平台,它定位于将产品快速且持续的部署到多种云平台上.Spinnaker 通过将发布和各个云平台解耦,来将 ...
- 【编程思想】【设计模式】【其他模式】graph_search
Python版 https://github.com/faif/python-patterns/blob/master/other/graph_search.py #!/usr/bin/env pyt ...
- linux基础-TCP/IP协议篇
一.网络TCP/IP层次模型 1.网络层次模型概念介绍:TCP/IP协议就是用于简化OSI层次,以及相关的标准.传输控制协议(tcp/ip)族是相关国防部(DoD)所创建的,主要用来确保数据的完整性及 ...
- 通过静态分析和持续集成 保证代码的质量 (Helix QAC)1
前言 现代软件开发团队面临着很多挑战,这些挑战包括:产品交付期限越来越紧,团队的分布越来越广,软件的复杂度越来越高,而且对软件的质量要求越来越高. 本文分为两个章节.第一章讨论持续集成的原理,持续集成 ...
- 阿里云发布CloudOps白皮书,ECS自动化运维套件新升级
12月10 日,2021云上架构与运维峰会上,阿里云发布业界首部<云上自动化运维白皮书>(简称CloudOps白皮书),并在其中提出了CloudOps成熟度模型.同时,阿里云还宣布了ECS ...
- GDB调试增强篇
GDB中应该知道的几个调试方法 七.八年前写过一篇<用GDB调试程序>, 于是,从那以后,很多朋友在MSN上以及给我发邮件询问我关于GDB的问题,一直到今天,还有人在问GDB的相关问题.这 ...