SQL 中 CASE WHEN 用法全面详解
CASE WHEN是 SQL 中非常强大的条件判断表达式,类似于编程语言中的if-else或switch语句。它可以根据条件返回不同的值,常用于数据转换、分组统计、动态排序等场景。
SQL 中有两种 CASE 语法:
1. 简单 CASE(类似 switch)
CASE表达式WHEN值1THEN结果1WHEN值2THEN结果2...ELSE默认结果END2. 搜索 CASE(更常用,类似 if-elseif-else)
CASEWHEN条件1THEN结果1WHEN条件2THEN结果2...ELSE默认结果END注意:END是必须的!否则报错。ELSE可省略,省略时相当于ELSE NULL。
一、基本用法示例
假设有表students:
| id | name | score |
|---|---|---|
| 1 | 张三 | 85 |
| 2 | 李四 | 62 |
| 3 | 王五 | 95 |
| 4 | 赵六 | 45 |
| 5 | null | null |
示例1:根据分数划分等级(搜索 CASE,最常用)
SELECTname,score,CASEWHENscore>=90THEN'优秀'WHENscore>=80THEN'良好'WHENscore>=60THEN'及格'ELSE'不及格'ENDASgradeFROMstudents;结果:
| name | score | grade |
|---|---|---|
| 张三 | 85 | 良好 |
| 李四 | 62 | 及格 |
| 王五 | 95 | 优秀 |
| 赵六 | 45 | 不及格 |
| null | null | 不及格(因为 NULL 不满足任何条件) |
示例2:简单 CASE(匹配具体值)
SELECTname,score,CASEscoreWHEN100THEN'满分'WHEN0THEN'零分'ELSE'普通分数'ENDASremarkFROMstudents;示例3:处理 NULL 值
SELECTname,CASEWHENscoreISNULLTHEN'未考试'WHENscore>=60THEN'通过'ELSE'未通过'ENDASstatusFROMstudents;示例4:在 ORDER BY 中动态排序(超级实用!)
-- 优先让 score=100 的排最前,其余按 score 降序SELECTname,scoreFROMstudentsORDERBYCASEWHENscore=100THEN0ELSE1END,-- 100分排前面scoreDESC;示例5:在 UPDATE 中使用(修改数据)
UPDATEstudentsSETgrade=CASEWHENscore>=90THEN'A'WHENscore>=80THEN'B'WHENscore>=60THEN'C'ELSE'F'END;示例6:结合聚合函数统计(分组统计神器)
SELECTCASEWHENscore>=90THEN'优秀'WHENscore>=60THEN'及格'ELSE'不及格'ENDASlevel,COUNT(*)AScount,AVG(score)ASavg_scoreFROMstudentsGROUPBYCASEWHENscore>=90THEN'优秀'WHENscore>=60THEN'及格'ELSE'不及格'END;结果示例:
| level | count | avg_score |
|---|---|---|
| 优秀 | 1 | 95.0 |
| 良好 | 1 | 85.0 |
| 及格 | 1 | 62.0 |
| 不及格 | 1 | 45.0 |
示例7:多条件组合(AND/OR)
CASEWHENscore>=90ANDnameLIKE'张%'THEN'优秀且姓张'WHENscore>=80ORscoreISNULLTHEN'良好或未考'ELSE'其他'END示例8:嵌套 CASE(不推荐过多嵌套,可读性差)
CASEWHENscoreISNULLTHEN'未考'ELSECASEWHENscore>=90THEN'优秀'ELSE'非优秀'ENDEND二、常见注意事项
| 注意点 | 说明 |
|---|---|
| 类型一致 | 所有THEN和ELSE的结果类型必须一致(都是字符串或数字) |
| NULL 判断 | 用IS NULL/IS NOT NULL,不能用= NULL |
| 条件执行顺序 | 从上到下,满足第一个条件就返回,后面的不再判断 |
| 可在 SELECT/WHERE/GROUP BY/ORDER BY/HAVING 中使用 | 全方位可用 |
| MySQL 支持在 WHERE 中直接用 CASE | 但建议逻辑放 SELECT 中更清晰 |
| 性能 | 过多复杂 CASE 可能影响性能,必要时考虑建视图或存储过程 |
三、实际应用场景总结
| 场景 | 推荐用法 |
|---|---|
| 数据清洗/转换 | 将代码转为可读文字(如 status 1→’启用’) |
| 动态分组统计 | 按条件分组 COUNT/SUM/AVG |
| 自定义排序规则 | ORDER BY CASE … |
| 报表字段生成 | 生成“等级”“是否达标”等衍生列 |
| 权限控制展示 | 根据角色显示不同内容 |
掌握CASE WHEN后,你的 SQL 查询能力会提升一个大台阶!它几乎是写复杂报表和数据分析 SQL 的必备武器。
如果你有具体业务场景(如“如何用 CASE WHEN 实现 pivot 表格”或“多表联查中的条件判断”),欢迎继续提问,我可以给出针对性示例!