SQL大师之路 15 条件分支

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


文章目录

    • [一、 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;

三、 使用要点

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

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

  3. 性能与索引

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


相关推荐
无关868818 分钟前
Redis Bitmaps 用户签到系统设计方案
数据库·redis·缓存
江华森26 分钟前
FastAPI 极速开发指南 — 从零到生产级 API 实战
数据库·fastapi
左直拳1 小时前
mysql分区表自动归档
mysql·分区表·分区表归档
MATLAB代码顾问1 小时前
Python Pandas数据分析入门指南
python·数据分析·pandas
老纪2 小时前
Redis分布式锁进第九零篇
数据库·redis·分布式
haven-8522 小时前
MySQL事务ACID、隔离级别、MVCC、幻读解决
数据库·mysql
小高学习java2 小时前
事务的边界问题,如何判断数据回滚时机。
java·数据库·后端
迷枫7123 小时前
【无标题】
数据库
TDengine (老段)3 小时前
TDengine 扫描算子 — TableScan、TagScan 与下推优化
大数据·数据库·物联网·时序数据库·tdengine·涛思数据
放下华子我只抽RuiKe53 小时前
FastAPI 全栈后端(三):数据库与 ORM
前端·数据库·react.js·oracle·性能优化·前端框架·fastapi