1. 执行子查询
SELECT 供应商号 FROM 订购单 WHERE 职工号 IN ('E1', 'E3') GROUP BY 供应商号 HAVING COUNT(DISTINCT 职工号) = 2
筛选职工号为
E1或E3的记录:
依据WHERE 职工号 IN ('E1', 'E3')这个条件,从订购单表中把职工号为E1或者E3的记录筛选出来。得到的结果如下:
| E3 | S7 | OR67 | 2002/06/23 |
| E1 | S4 | OR73 | 2002/07/28 |
| E3 | S4 | OR79 | 2002/06/13 |
| E3 | S3 | OR91 | 2002/07/13 |按
供应商号分组:
运用GROUP BY 供应商号对筛选出来的结果进行分组,分组情况如下:
- 供应商号
S3:包含记录(E3, S3, OR91, 2002/07/13)- 供应商号
S4:包含记录(E1, S4, OR73, 2002/07/28)和(E3, S4, OR79, 2002/06/13)- 供应商号
S7:包含记录(E3, S7, OR67, 2002/06/23)使用
HAVING子句筛选分组结果:
HAVING COUNT(DISTINCT 职工号) = 2的作用是只保留分组中不同职工号数量为 2 的分组。
- 对于供应商号
S3,其分组里只有职工号E3,不同职工号数量为 1,不满足条件。- 对于供应商号
S4,分组中有职工号E1和E3,不同职工号数量为 2,符合条件。- 对于供应商号
S7,分组里只有职工号E3,不同职工号数量为 1,不符合条件。
所以,子查询最终返回的供应商号是S4。2. 执行主查询
SELECT * FROM 供应商 WHERE 地址 = '北京' AND 供应商号 IN ('S4')
筛选地址为北京的供应商:
根据WHERE 地址 = '北京'条件,从供应商表中筛选出地址为北京的供应商记录。结果如下:
| 供应商号 | 供应商名 | 地址 |
| S4 | 华通电子公司 | 北京 |
| S7 | 爱华电子厂 | 北京 |筛选
供应商号在子查询结果中的记录:
按照AND 供应商号 IN ('S4')条件,从上述筛选结果中进一步筛选出供应商号为S4的记录。最终结果如下:
| 供应商号 | 供应商名 | 地址 |
| S4 | 华通电子公司 | 北京 |
1. 子查询部分
SELECT 仓库号, AVG(工资) AS avg_salary FROM 职工 GROUP BY 仓库号
GROUP BY 仓库号:按照仓库号对职工表中的记录进行分组。也就是说,会把在同一个仓库工作的职工记录分为一组。SELECT 仓库号, AVG(工资) AS avg_salary:对于每一组,计算该组内职工工资的平均值,并将结果命名为avg_salary,同时选择仓库号作为分组标识。例如,如果有两个仓库WH1和WH2,那么会分别计算WH1仓库和WH2仓库中职工的平均工资。这个子查询的结果是一个临时表,包含两列:
仓库号和该仓库对应的avg_salary。2.
JOIN ON部分JOIN ( -- 子查询 SELECT 仓库号, AVG(工资) AS avg_salary FROM 职工 GROUP BY 仓库号 ) s ON t.仓库号 = s.仓库号
JOIN:JOIN是用于将两个或多个表连接起来的操作符。在这里,将职工表(用别名t表示)和子查询结果(用别名s表示)进行连接。(...) s:子查询被当作一个临时表,使用别名s来引用它。ON t.仓库号 = s.仓库号:ON关键字后面的条件指定了连接的规则。这里表示将职工表中的仓库号与子查询结果中的仓库号进行匹配,只有当两者相等时,对应的记录才会被连接在一起。例如,如果职工表中有一条记录的仓库号是WH1,子查询结果中也有WH1对应的平均工资记录,那么这两条记录就会被连接起来。3. 主查询和
WHERE部分SELECT t.* FROM 职工 t JOIN ( SELECT 仓库号, AVG(工资) AS avg_salary FROM 职工 GROUP BY 仓库号 ) s ON t.仓库号 = s.仓库号 WHERE t.工资 < s.avg_salary;
SELECT t.*:选择职工表(别名t)中的所有列。WHERE t.工资 < s.avg_salary:WHERE子句用于筛选满足条件的记录。这里表示只选择职工表中工资小于其所在仓库平均工资的记录。示例说明
假设
职工表数据如下:
仓库号 职工号 工资 WH1 E1 2000 WH1 E2 3000 WH2 E3 4000 WH2 E4 5000 子查询计算出的平均工资结果如下:
仓库号 avg_salary WH1 2500 WH2 4500 通过
JOIN ON连接后,会将职工表和子查询结果进行匹配,得到:
仓库号 职工号 工资 avg_salary WH1 E1 2000 2500 WH1 E2 3000 2500 WH2 E3 4000 4500 WH2 E4 5000 4500 最后,通过
WHERE子句筛选出工资小于平均工资的记录,即:
仓库号 职工号 工资 avg_salary WH1 E1 2000 2500 WH2 E3 4000 4500 这样就得到了工资低于所在仓库平均工资的职工信息。
FROM 职工:指定从职工表中获取数据。
(7)检索学习全部课程的学生姓名

我们还是基于之前的示例数据来详细分析这个 SQL 语句每一步的执行结果。下面是涉及的三个表数据:
学生表
S
SNO SNAME AGE SEX SDEPT S1 张三 20 男 计算机系 S2 李四 21 女 数学系 S3 王五 20 男 计算机系 课程表
C
CNO CNAME CDEPT TNAME C1 数据库原理 计算机系 赵老师 C2 高等数学 数学系 钱老师 C3 编程语言 计算机系 孙老师 学生选课表
SC
SNO CNO GRADE S1 C1 85 S1 C3 90 S2 C2 78 S3 C1 88 S3 C2 92 S3 C3 80 1. 最内层子查询(针对每个学生和课程组合)
最内层子查询为
SELECT * FROM SC sc WHERE sc.SNO = s.SNO AND sc.CNO = c.CNO,它会针对每个学生和每门课程的组合进行查询,判断该学生是否选修了这门课程。对于学生
S1
- 当课程为
C1时,查询结果:
| SNO | CNO | GRADE |
| ---- | ---- | ---- |
| S1 | C1 | 85 |- 当课程为
C2时,由于S1未选修C2,查询结果为空表。- 当课程为
C3时,查询结果:
| SNO | CNO | GRADE |
| ---- | ---- | ---- |
| S1 | C3 | 90 |对于学生
S2
- 当课程为
C1时,由于S2未选修C1,查询结果为空表。- 当课程为
C2时,查询结果:
| SNO | CNO | GRADE |
| ---- | ---- | ---- |
| S2 | C2 | 78 |- 当课程为
C3时,由于S2未选修C3,查询结果为空表。对于学生
S3
- 当课程为
C1时,查询结果:
| SNO | CNO | GRADE |
| ---- | ---- | ---- |
| S3 | C1 | 88 |- 当课程为
C2时,查询结果:
| SNO | CNO | GRADE |
| ---- | ---- | ---- |
| S3 | C2 | 92 |- 当课程为
C3时,查询结果:
| SNO | CNO | GRADE |
| ---- | ---- | ---- |
| S3 | C3 | 80 |2. 中间层子查询(针对每个学生)
中间层子查询为
SELECT * FROM C c WHERE NOT EXISTS (SELECT * FROM SC sc WHERE sc.SNO = s.SNO AND sc.CNO = c.CNO),它会针对每个学生,找出该学生未选修的课程。对于学生
S1因为
S1未选修C2,所以查询结果:
CNO CNAME CDEPT TNAME C2 高等数学 数学系 钱老师 对于学生
S2因为
S2未选修C1和C3,所以查询结果:
CNO CNAME CDEPT TNAME C1 数据库原理 计算机系 赵老师 C3 编程语言 计算机系 孙老师 对于学生
S3由于
S3选修了所有课程,中间层子查询结果为空表。3. 最外层查询
最外层查询为
SELECT SNAME FROM S s WHERE NOT EXISTS (SELECT * FROM C c WHERE NOT EXISTS (SELECT * FROM SC sc WHERE sc.SNO = s.SNO AND sc.CNO = c.CNO)),它会筛选出选修了所有课程的学生姓名。
- 对于学生
S1和S2,中间层子查询有结果返回,NOT EXISTS为FALSE,不会被选中。- 对于学生
S3,中间层子查询结果为空表,NOT EXISTS为TRUE,会被选中。最终的查询结果为:
SNAME 王五
(8)查询所学课程包含学生 S3 所学课程的学生学号

学生选课表
SC
SNO CNO GRADE S1 C1 85 S1 C3 90 S2 C2 78 S3 C1 88 S3 C2 92 S3 C3 80 语句详解
- 最外层查询:
SELECT DISTINCT SNO FROM SC s1 WHERE...从学生选课表SC中选择不同的学生学号SNO,表SC使用别名s1,并且后面跟着筛选条件。- 中间层子查询:
SELECT 1 FROM SC s2 WHERE s2.SNO = 'S3' AND NOT EXISTS (...)从学生选课表SC(使用别名s2)中筛选出学号为S3的记录,并且对于每一条这样的记录,再嵌套一个子查询进行进一步判断。- 最内层子查询:
SELECT 1 FROM SC s3 WHERE s3.SNO = s1.SNO AND s3.CNO = s2.CNO从学生选课表SC(使用别名s3)中查找是否存在满足学号等于外层s1的学号且课程号等于中间层s2的课程号的记录。如果存在,该子查询返回一行(值为1),否则不返回行。整体逻辑是:对于
SC表中的每一个学生s1,检查S3选修的每一门课程,看s1是否也选修了这门课程。如果S3选修的所有课程s1都选修了,那么s1就满足条件,会被选入最终结果。每一步结果
初始化外层循环,遍历
SC表中的每一个学生(以s1表示)
- 当
s1.SNO = S1时:
- 中间层子查询:
SELECT 1 FROM SC s2 WHERE s2.SNO = 'S3',得到S3选修课程的记录:
| SNO | CNO | GRADE |
| ---- | ---- | ---- |
| S3 | C1 | 88 |
| S3 | C2 | 92 |
| S3 | C3 | 80 |- 对于上述每一条记录进行最内层子查询:
- 当
s2.CNO = C1时,SELECT 1 FROM SC s3 WHERE s3.SNO = 'S1' AND s3.CNO = 'C1',有记录,返回一行(值为1)。- 当
s2.CNO = C2时,SELECT 1 FROM SC s3 WHERE s3.SNO = 'S1' AND s3.CNO = 'C2',无记录,不返回行。- 当
s2.CNO = C3时,SELECT 1 FROM SC s3 WHERE s3.SNO = 'S1' AND s3.CNO = 'C3',有记录,返回一行(值为1)。- 因为存在
S3选修的课程(C2)而S1未选修,所以中间层子查询的NOT EXISTS部分对于S1为FALSE,S1不满足最外层查询条件。- 当
s1.SNO = S2时:
- 中间层子查询:
SELECT 1 FROM SC s2 WHERE s2.SNO = 'S3',得到S3选修课程的记录(同上面S3的选修课程记录)。- 对于上述每一条记录进行最内层子查询:
- 当
s2.CNO = C1时,SELECT 1 FROM SC s3 WHERE s3.SNO = 'S2' AND s3.CNO = 'C1',无记录,不返回行。- 当
s2.CNO = C2时,SELECT 1 FROM SC s3 WHERE s3.SNO = 'S2' AND s3.CNO = 'C2',有记录,返回一行(值为1)。- 当
s2.CNO = C3时,SELECT 1 FROM SC s3 WHERE s3.SNO = 'S2' AND s3.CNO = 'C3',无记录,不返回行。- 因为存在
S3选修的课程(C1和C3)而S2未选修,所以中间层子查询的NOT EXISTS部分对于S2为FALSE,S2不满足最外层查询条件。- 当
s1.SNO = S3时:
- 中间层子查询:
SELECT 1 FROM SC s2 WHERE s2.SNO = 'S3',得到S3选修课程的记录(同上面S3的选修课程记录)。- 对于上述每一条记录进行最内层子查询:
- 当
s2.CNO = C1时,SELECT 1 FROM SC s3 WHERE s3.SNO = 'S3' AND s3.CNO = 'C1',有记录,返回一行(值为1)。- 当
s2.CNO = C2时,SELECT 1 FROM SC s3 WHERE s3.SNO = 'S3' AND s3.CNO = 'C2',有记录,返回一行(值为1)。- 当
s2.CNO = C3时,SELECT 1 FROM SC s3 WHERE s3.SNO = 'S3' AND s3.CNO = 'C3',有记录,返回一行(值为1)。- 因为
S3选修的所有课程S3自己都选修了,所以中间层子查询的NOT EXISTS部分对于S3为TRUE,S3满足最外层查询条件。最终结果:
经过最外层的DISTINCT处理后,得到满足条件的学生学号:
| S3 |


















