SQL实战:01之行转列实现

文章目录

概述

我们在工作中遇到的很多场景需要将数据表中的一行的值转为一列的值,为实现这种场景可以通过IF函数或者CASE WHEN的方式来实现。恰好本人最近在刷题,就以碰到的LeetCode中的题 为例子来讲解如何使用IF和 CASE WHEN 实现行转列。

语法

IF的使用语法

IF是一个函数,有三个参数,第一个参数是判断条件,第二个参数是条件成立时的取值,第三个参数是条件不成立时的取值。

SQL 复制代码
IF(条件, 值1, 值2 )

除此以外,IF还可以嵌套使用,在一些复杂的多分支场景中就可以通过嵌套的方式来实现。

SQL 复制代码
IF(条件, 值1, IF( 条件, 值1, 值2 ) )

CASE WHEN 使用语法

SQL 复制代码
CASE WHEN 条件  THEN 操作
     WHEN 条件  THEN 操作
     ......
     ELSE 操作
END

和IF一样,CASE WHEN 也可以嵌套使用,但是得注意,每个CASE WHEN都需要保持完整。

SQL 复制代码
CASE WHEN 条件  THEN 操作
     WHEN 条件  THEN 操作
     ......
     ELSE 
        CASE WHEN 子条件 THEN 操作
             ELSE 操作 
        END
END

行转列场景

题目:重新格式话部门表

复制代码
表 Department:

+---------------+---------+
| Column Name   | Type    |
+---------------+---------+
| id            | int     |
| revenue       | int     |
| month         | varchar |
+---------------+---------+
在 SQL 中,(id, month) 是表的联合主键。
这个表格有关于每个部门每月收入的信息。
月份(month)可以取下列值 ["Jan","Feb","Mar","Apr","May","Jun","Jul","Aug","Sep","Oct","Nov","Dec"]。
 

重新格式化表格,使得 每个月 都有一个部门 id 列和一个收入列。

以 任意顺序 返回结果表。

结果格式如以下示例所示。

 

示例 1:

输入:
Department table:
+------+---------+-------+
| id   | revenue | month |
+------+---------+-------+
| 1    | 8000    | Jan   |
| 2    | 9000    | Jan   |
| 3    | 10000   | Feb   |
| 1    | 7000    | Feb   |
| 1    | 6000    | Mar   |
+------+---------+-------+
输出:
+------+-------------+-------------+-------------+-----+-------------+
| id   | Jan_Revenue | Feb_Revenue | Mar_Revenue | ... | Dec_Revenue |
+------+-------------+-------------+-------------+-----+-------------+
| 1    | 8000        | 7000        | 6000        | ... | null        |
| 2    | 9000        | null        | null        | ... | null        |
| 3    | null        | 10000       | null        | ... | null        |
+------+-------------+-------------+-------------+-----+-------------+
解释:四月到十二月的收入为空。 
请注意,结果表共有 13 列(1 列用于部门 ID,其余 12 列用于各个月份)。

题解

  • 使用IF函数实现
sql 复制代码
select id
    ,SUM(IF(month='Jan',revenue,NULL)) AS Jan_Revenue
    ,SUM(IF(month='Feb',revenue,NULL)) AS Feb_Revenue
    ,SUM(IF(month='Mar',revenue,NULL)) AS Mar_Revenue
    ,SUM(IF(month='Apr',revenue,NULL)) AS Apr_Revenue
    ,SUM(IF(month='May',revenue,NULL)) AS May_Revenue
    ,SUM(IF(month='Jun',revenue,NULL)) AS Jun_Revenue
    ,SUM(IF(month='Jul',revenue,NULL)) AS Jul_Revenue
    ,SUM(IF(month='Aug',revenue,NULL)) AS Aug_Revenue
    ,SUM(IF(month='Sep',revenue,NULL)) AS Sep_Revenue
    ,SUM(IF(month='Oct',revenue,NULL)) AS Oct_Revenue
    ,SUM(IF(month='Nov',revenue,NULL)) AS Nov_Revenue
    ,SUM(IF(month='Dec',revenue,NULL)) AS Dec_Revenue
from Department
group by id;
  • 使用CASE WHEN实现
sql 复制代码
select id
    ,SUM(CASE WHEN month='Jan' THEN revenue ELSE NULL END) AS Jan_Revenue
    ,SUM(CASE WHEN month='Feb' THEN revenue ELSE NULL END) AS Feb_Revenue
    ,SUM(CASE WHEN month='Mar' THEN revenue ELSE NULL END) AS Mar_Revenue
    ,SUM(CASE WHEN month='Apr' THEN revenue ELSE NULL END) AS Apr_Revenue
    ,SUM(CASE WHEN month='May' THEN revenue ELSE NULL END) AS May_Revenue
    ,SUM(CASE WHEN month='Jun' THEN revenue ELSE NULL END) AS Jun_Revenue
    ,SUM(CASE WHEN month='Jul' THEN revenue ELSE NULL END) AS Jul_Revenue
    ,SUM(CASE WHEN month='Aug' THEN revenue ELSE NULL END) AS Aug_Revenue
    ,SUM(CASE WHEN month='Sep' THEN revenue ELSE NULL END) AS Sep_Revenue
    ,SUM(CASE WHEN month='Oct' THEN revenue ELSE NULL END) AS Oct_Revenue
    ,SUM(CASE WHEN month='Nov' THEN revenue ELSE NULL END) AS Nov_Revenue
    ,SUM(CASE WHEN month='Dec' THEN revenue ELSE NULL END) AS Dec_Revenue
from Department
group by id;

由以上两种实现方式可以看出,IF 比 CASE WHEN的实现方式看起来更加简洁,所以如果可以使用IF函数来实现的,都可以采用IF函数来实现,避免代码的臃肿。

相关推荐
言德斐13 小时前
SQL性能优化的思路及策略
数据库·sql·性能优化
码界奇点13 小时前
Django视图从基础到高级的全面解析
数据库·django·sqlite·web·python3.11
Allan_202513 小时前
数据库学习
数据库·学习
fen_fen13 小时前
人大金仓数据库kingbase8创建表示例
数据库·oracle
一勺菠萝丶13 小时前
「您的连接不是私密连接」详解:为什么 HTTPS 证书会报错,以及如何正确配置子域名证书
数据库·网络协议·https
²º²²এ松14 小时前
蓝牙低功耗(BLE)通信的中心设备/外围设备(连接角色)、主机/从机(时序角色)、客户端/服务器(数据交互角色)的理解
运维·服务器·数据库
百锦再14 小时前
Vue Scoped样式混淆问题详解与解决方案
java·前端·javascript·数据库·vue.js·学习·.net
数据库知识分享者小北15 小时前
云栖重磅|瑶池数据库:从云原生数据底座向“AI就绪”的多模态数据底座演进
数据库·人工智能·云原生
_Johnny_15 小时前
Redis 升级操作指南:单机与主从模式
数据库·redis·缓存
源力祁老师15 小时前
ODOO数据文件(XML、CSV、SQL)是如何转换并加载到 Odoo 数据库
xml·数据库·sql