SQL大师之路 15 条件分支

在数据库查询中,我们经常会遇到需要根据不同条件返回不同结果的场景,虽然这些逻辑可以在应用层处理,但在数据库层面直接处理通常能减少应用层与数据库的交互次数,合并多次查询为一次,从而显著提高执行效率。


一、 CASE 表达式

CASE 是 MySQL的分支判断语句,它遵循标准 SQL 规范,具有极佳的通用性。下面用employees库演示其用法。

1.1 简单 CASE 模式

当需要进行简单的“等值匹配”时,类似于编程语言中的 switch-case。字段会依次与分支中的值进行比较。

语法:其中ELSE语句是可选的,如果没有匹配的分支同时没有ELSE语句,将返回null

CASE column_name
    WHEN value1 THEN result1
    WHEN value2 THEN result2
    ELSE default_result
END

案例:在 dept_emp 表中,部门编号如 d001, d005 并不直观,通过 CASE 我们可以快速将其映射为中文名称。

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 以及各种逻辑运算符。

语法

CASE 
    WHEN condition1 THEN result1
    WHEN condition2 THEN result2
    ELSE default_result
END

案例:根据入职年份确定奖金系数,如果入职早于1990或者职位是’Senior Engineer’/‘Manager’ 取最高比例。

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”)转换为女/男。

SELECT 
    concat(last_name, ' ', first_name) 姓名,
    IF(gender='F', '女', '男') 性别
FROM employees
LIMIT 10;

在这里插入图片描述

三、 使用要点

  1. 短路特性:CASE 会按顺序评估 WHEN 子句。一旦某个条件为真,它就会返回对应结果并停止后续评估。

  2. NULL 值的陷阱:如果在 WHEN 条件中涉及可能为 NULL 的字段,务必结合 IS NULLIFNULL() 处理,否则 UNKNOWN 的判断结果会导致该条数据直接滑向 ELSE 分支。

  3. 性能与索引
    尽量在 SELECT 子句中使用,如果在 WHERE 子句中使用会导致数据库无法利用索引进行扫描(和函数类似),在大数据量库中会导致性能问题。


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值