3 种高性能写法,专门解决:
查询【学过 001 号同学所有课程】的学生
(比双层 NOT EXISTS 更高效、更现代、大厂常用)
🔥 写法 1:GROUP BY + HAVING(最高性能、最简洁)
这是大数据量下最快的写法,面试 / 工作首选。
sql
SELECT s.s_no, s.sname
FROM Student s
JOIN Sc sc ON s.s_no = sc.s_no
-- 只查001选过的课程
WHERE sc.c_no IN (SELECT c_no FROM Sc WHERE s_no = '001')
AND s.s_no <> '001'
GROUP BY s.s_no, s.sname
-- 关键:选的课程数 = 001选的总课程数
HAVING COUNT(DISTINCT sc.c_no) = (
SELECT COUNT(DISTINCT c_no) FROM Sc WHERE s_no = '001'
);
原理(超简单)
- 先把学生只限制在 001 选过的课程里
- 统计每个学生选了几门
- 数量 = 001 选的总课数 → 就是全选了
🔥 写法 2:JOIN 分组匹配(极高速)
适合超大数据量表,性能碾压 NOT EXISTS
sql
SELECT s.s_no, s.sname
FROM Student s
JOIN (
SELECT sc2.s_no
FROM Sc sc1
JOIN Sc sc2 ON sc1.c_no = sc2.c_no
WHERE sc1.s_no = '001'
GROUP BY sc2.s_no
HAVING COUNT(DISTINCT sc2.c_no) = (
SELECT COUNT(DISTINCT c_no) FROM Sc WHERE s_no = '001'
)
) t ON s.s_no = t.s_no
WHERE s.s_no <> '001';
🔥 写法 3:WITH 子句(现代 SQL,可读性最强)
MySQL 8.0 / PostgreSQL / SQL Server 通用
sql
WITH Course001 AS (
SELECT DISTINCT c_no FROM Sc WHERE s_no = '001'
),
TotalCourse AS (
SELECT COUNT(*) AS total FROM Course001
)
SELECT s.s_no, s.sname
FROM Student s
JOIN Sc sc ON s.s_no = sc.s_no
JOIN Course001 c ON sc.c_no = c.c_no
WHERE s.s_no <> '001'
GROUP BY s.s_no, s.sname
HAVING COUNT(DISTINCT sc.c_no) = (SELECT total FROM TotalCourse);
🚀 三种写法性能对比(直接告诉你用哪个)
表格
| 写法 | 性能 | 适用场景 | 推荐指数 |
|---|---|---|---|
| GROUP BY + HAVING | ⭐⭐⭐⭐⭐ | 所有场景,首选 | 最强 |
| JOIN 分组 | ⭐⭐⭐⭐⭐ | 千万级大数据 | 极强 |
| WITH 子句 | ⭐⭐⭐⭐ | 现代 SQL、代码清晰 | 优秀 |
| NOT EXISTS | ⭐⭐ | 教学、小数据 | 一般 |
✅ 总结一句话
查询 "选了全部 XX 课程",别用 NOT EXISTS,用 GROUP BY + HAVING 性能最高!