深入浅出:SQL Server 中的 PIVOT 与 UNPIVOT 用法详解

在这里插入图片描述

引言

在数据分析与报表生成场景中,行列转换是一个高频需求。SQL Server 提供了 PIVOTUNPIVOT 两个强大的运算符,能够帮助我们快速实现数据透视与逆透视操作。本文将结合具体示例,解析它们的核心用法。

一、PIVOT:将行转换为列

PIVOT函数主要是用来将数据从行转换成列。比如,如果有订单数据表,里面有很多订单的信息,可能按客户ID、订单日期等分组。使用PIVOT可以把这些重复的客户信息排列成一个更紧凑的表格,每个客户的订单日期变成一列,这样看起来更直观。

核心作用

将某一列的唯一值作为新列名,并按需聚合关联数据。

语法结构

SELECT [非透视列], [透视列1], [透视列2], ...
FROM (
    SELECT [1], [2], [聚合列] 
    FROM) AS 源表
PIVOT (
    聚合函数(聚合列)
    FOR [目标列] IN ([透视值1], [透视值2], ...)
) AS 别名;

实战示例

场景:统计各部门在不同季度的销售额。

  1. 准备数据
CREATE TABLE #Sales (
    Department VARCHAR(50),
    Quarter CHAR(2),
    Amount DECIMAL(10,2)
);

INSERT INTO #Sales VALUES
('HR', 'Q1', 20000),
('HR', 'Q2', 22000),
('IT', 'Q1', 35000),
('IT', 'Q3', 41000);
  1. 执行 PIVOT
SELECT Department, [Q1], [Q2], [Q3], [Q4]
FROM (
    SELECT Department, Quarter, Amount 
    FROM #Sales
) AS Src
PIVOT (
    SUM(Amount)
    FOR Quarter IN ([Q1], [Q2], [Q3], [Q4])
) AS Pvt;

输出结果

在这里插入图片描述

二、UNPIVOT:将列转换为行

UNPIVOT函数,它的作用和PIVOT相反,是用来把数据从列转换回行。比如,在PIVOT之后得到的一张表格里,如果需要进一步细分数据或者进行其他操作,可以用UNPIVOT来恢复原来的多行结构。

核心作用

将多列合并为两列(属性名+属性值),实现数据逆向透视。

语法结构

SELECT [非透视列], [属性列], [值列]
FROMUNPIVOT (
    值列 FOR 属性列 IN ([1], [2], ...)
) AS 别名;

实战示例

场景:将季度销售额列还原为行结构。

  1. 使用之前 PIVOT 的结果作为输入
CREATE TABLE #PivotedSales (
    Department VARCHAR(50),
    Q1 DECIMAL(10,2),
    Q2 DECIMAL(10,2),
    Q3 DECIMAL(10,2),
    Q4 DECIMAL(10,2)
);

INSERT INTO #PivotedSales VALUES
('HR', 20000, 22000, NULL, NULL),
('IT', 35000, NULL, 41000, NULL);
  1. 执行 UNPIVOT
SELECT Department, Quarter, Amount
FROM #PivotedSales
UNPIVOT (
    Amount FOR Quarter IN (Q1, Q2, Q3, Q4)
) AS Unpvt;

输出结果
在这里插入图片描述

三、关键注意事项

  1. 数据类型一致性
    UNPIVOT 的所有列必须具有兼容的数据类型。

  2. 处理 NULL 值
    PIVOT 会自动过滤 NULL 值,可通过 ISNULL()COALESCE() 预处理。

  3. 动态列处理
    当透视列值不固定时,需使用动态 SQL 拼接列名(示例需另写代码实现)。

  4. 性能优化
    对大型数据集建议建立合适索引,避免全表扫描。

四、典型应用场景对比

操作适用场景示例
PIVOT生成交叉报表、统计类报表部门季度销售汇总
UNPIVOT数据规范化、ETL预处理、存储优化将多个月份列合并为日期维度

五、总结

  • PIVOT 通过聚合实现行转列,适合制作汇总视图
  • UNPIVOT 通过逆向操作恢复数据结构,适合数据清洗
  • 二者配合使用可完成复杂数据转换需求
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

dotnet研习社

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值