在数据库查询中,我们经常会遇到需要根据不同条件返回不同结果的场景,虽然这些逻辑可以在应用层处理,但在数据库层面直接处理通常能减少应用层与数据库的交互次数,合并多次查询为一次,从而显著提高执行效率。
文章目录
-
- [一、 CASE 表达式](#一、 CASE 表达式)
-
- [1.1 简单 CASE 模式](#1.1 简单 CASE 模式)
- [1.2 搜索 CASE 模式](#1.2 搜索 CASE 模式)
- [二、 IF 函数](#二、 IF 函数)
- [三、 使用要点](#三、 使用要点)
一、 CASE 表达式
CASE 是 MySQL的分支判断语句,它遵循标准 SQL 规范,具有极佳的通用性。下面用employees库演示其用法。
1.1 简单 CASE 模式
当需要进行简单的"等值匹配"时,类似于编程语言中的 switch-case。字段会依次与分支中的值进行比较。
语法:其中ELSE语句是可选的,如果没有匹配的分支同时没有ELSE语句,将返回null
sql
CASE column_name
WHEN value1 THEN result1
WHEN value2 THEN result2
ELSE default_result
END
案例:在 dept_emp 表中,部门编号如 d001, d005 并不直观,通过 CASE 我们可以快速将其映射为中文名称。
sql
SELECT dept_no 部门编号,
CASE dept_no
WHEN 'd001' THEN '市场部'
WHEN 'd002' THEN '财务部'
WHEN 'd003' THEN '人力资源'
WHEN 'd004' THEN '开发部'
ELSE '其他部门'
END 部门名称
FROM departments
LIMIT 10;

1.2 搜索 CASE 模式
搜索型 CASE 的强大之处在于它不局限于等值匹配,可以组合 AND、OR 以及各种逻辑运算符。
语法:
sql
CASE
WHEN condition1 THEN result1
WHEN condition2 THEN result2
ELSE default_result
END
案例:根据入职年份确定奖金系数,如果入职早于1990或者职位是'Senior Engineer'/'Manager' 取最高比例。
sql
SELECT
e.emp_no,
e.first_name,
e.last_name,
CASE
WHEN e.hire_date < '1990-01-01' OR t.title IN ('Senior Engineer', 'Manager') THEN 0.8
WHEN e.hire_date < '1995-01-01' THEN 0.6
ELSE 0.4
END 奖金系数
FROM employees e
JOIN titles t ON e.emp_no = t.emp_no;

二、 IF 函数
IF() 是 轻量级的三元运算符,非常适合处理 SQL 查询中简单的"二选一"逻辑。
语法
IF(expr1, expr2, expr3):如果 expr1 为 TRUE 返回 expr2,否则返回 expr3。
案例:将employees表中的gender(包含值"F","M")转换为女/男。
sql
SELECT
concat(last_name, ' ', first_name) 姓名,
IF(gender='F', '女', '男') 性别
FROM employees
LIMIT 10;

三、 使用要点
-
短路特性 :CASE 会按顺序评估
WHEN子句。一旦某个条件为真,它就会返回对应结果并停止后续评估。 -
NULL 值的陷阱 :如果在
WHEN条件中涉及可能为NULL的字段,务必结合IS NULL或IFNULL()处理,否则UNKNOWN的判断结果会导致该条数据直接滑向ELSE分支。 -
性能与索引 :
尽量在
SELECT子句中使用,如果在WHERE子句中使用会导致数据库无法利用索引进行扫描(和函数类似),在大数据量库中会导致性能问题。