SQL server 2005 UNPIVOT运算符的使用

2023-10-10 22:18

本文主要是介绍SQL server 2005 UNPIVOT运算符的使用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

      UNPIVOT运算符相对于PIVOT运算符,它执行与PIVOT相反的操作,即将列转换到行。需要注意的是UNPIVOT运算符并不完全是PIVOT的逆向操作。因为执行PIVOT将会执行一次聚合,从而将多个可能的行合并为输出中的单个行。而 UNPIVOT 无法重现原始表值表达式的结果,因为行已经被合并了。另外,UNPIVOT 的输入中的空值不会显示在输出中,而在执行 PIVOT 操作之前,输入中可能有原始的空值。

      关于PIVOT运算符的使用,请参考:SQL server 2005 PIVOT运算符的使用

一、UNPIVOT的语法

SELECT
[DataType]
,   [DataValue]
,   [UnpivotedCol1]
,   [UnpivotedCol2]
,   [UnpivotedCol3]
FROM table
UNPIVOT
(
[DataValue] FOR [DataType]
IN ([UnpivotedCol1],[UnpivotedCol1],[UnpivotedCol3],etc..)
)
AS Alias
--需要注意的是,对于每一个UNPIVOT列值,必须使用中括号括起来。
二、UNPIVOT的使用例子

     我们使用了SQL server 2005 PIVOT运算符的使用文章中的例子。如下所示,从NorthWind数据库生成数据源及PIVOT表。

1.静态UNPIVOT的用法

    USE tempdb
GO
SELECT YEAR(OrderDate) AS [Year]  --从NorthWind提取演示示例所需数据
,CustomerID 
,od.Quantity
INTO dbo.Orders       
FROM NorthWind..Orders AS o
JOIN NorthWind..[Order Details] AS od
ON o.OrderID = od.OrderID
WHERE o.CustomerID IN ('BONAP','BOTTM','ANTON')
AND od.Quantity > 20;
SELECT * FROM dbo.Orders;
/*以下列出部分结果集,年的数据省略
Year        CustomerID Quantity
----------- ---------- --------
1996        BONAP      40
1996        BONAP      50
1996        BONAP      24
1996        ANTON      24
1996        BOTTM      30
1997        BOTTM      49
1997        BOTTM      25
1997        BOTTM      40
1997        BOTTM      50
1997        BOTTM      50
1997        BOTTM      30
*/
--1->.生成PIVOT数据
SELECT CustomerID
,[1996],[1997],[1998]
INTO #pivot_result	
FROM dbo.Orders
PIVOT (
SUM(Quantity)
FOR [Year] IN ([1996],[1997],[1998])
)x
/* Result:   
CustomerID 1996        1997        1998
---------- ----------- ----------- -----------
ANTON      24          205         NULL
BONAP      114         356         122
BOTTM      30          389         292
*/
--2->.生成UNPIVOT数据,实现列到行的转换
SELECT CustomerID
,Year
,Quantity
FROM #pivot_result 
UNPIVOT 
(
Quantity FOR Year IN ([1996],[1997],[1998])
)  x    
/* Result:
CustomerID Year        Quantity
---------- ----------- -----------
ANTON      1996        24
ANTON      1997        205
BONAP      1996        114
BONAP      1997        356
BONAP      1998        122
BOTTM      1996        30
BOTTM      1997        389
BOTTM      1998        292
*/
--从上面的结果中我们看出CustomerID为ANTON用户在年,Quantity的值为NULL,即没有订单,使用UNPIOVT后忽略掉了NULL值。
--注意UNPIVOT后并没有回到PIVOT之前的数据,因为实现PIVOT后数据已经被汇总。
--3->.在UNPIVOT后的结果中重现NULL值
SELECT CustomerID
,Year
,CASE WHEN Quantity = 0 THEN NULL ELSE Quantity END AS Quantity --还原为NULL的值
FROM (
SELECT CustomerID,Year,Quantity
FROM (        
SELECT CustomerID,
ISNULL([1996],0) AS [1996] --对年份列增加了一个ISNULL的判断
,ISNULL([1997],0) AS [1997]
,ISNULL([1998],0) AS [1998]
FROM #pivot_result) AS  pivot_result 
UNPIVOT 
(
Quantity FOR Year IN ([1996],[1997],[1998])
)  x      
)y
/*
CustomerID Year    Quantity
---------- ------- -----------
ANTON      1996     24
ANTON      1997     205
ANTON      1998     NULL  --NULL值重现
BONAP      1996     114
BONAP      1997     356
BONAP      1998     122
BOTTM      1996     30
BOTTM      1997     389
BOTTM      1998     292
*/

2.动态UNPIVOT的用法

 USE tempd;GO DECLARE @UnpivotColHeader NVARCHAR(MAX),@UnpivotTableSQL NVARCHAR(MAX);SELECT @UnpivotColHeader =  --将列标题转换为行值STUFF((SELECT N','+ QUOTENAME(x) AS [text()]FROM (SELECT COLUMN_NAME AS xFROM INFORMATION_SCHEMA.COLUMNSWHERE TABLE_SCHEMA = N'dbo'AND TABLE_NAME = N'pivot_result'AND COLUMN_NAME NOT IN(N'CustomerID')) AS xORDER BY xFOR XML PATH('')),   1,1,N''); SET @UnpivotTableSQL ='SELECT CustomerID,Year,CASE WHEN Quantity = 0 THEN NULL ELSE Quantity END AS Quantity FROM (SELECT CustomerID,Year,QuantityFROM (        SELECT CustomerID,ISNULL([1996],0) AS [1996] ,ISNULL([1997],0) AS [1997],ISNULL([1998],0) AS [1998]FROM pivot_result) AS  pivot_result UNPIVOT (Quantity FOR Year IN (' + @UnpivotColHeader + '))  x      )y' + N';';EXEC sp_executesql @UnpivotTableSQL;--结果同上


三、总结

1.UNPIVOT操作符的第一个输入是保存被旋转属性值(Quantity)的目标列名称。

2.FOR关键字,指定保存被旋转列名称(Year)的目标列名称。

3.在IN子句的圆括号,指定要旋转的源列名称([1996],[1997],[1998])。

 

PIVOT运算符的使用,请参阅:http://blog.csdn.net/robinson_0612/archive/2010/03/16/5385117.aspx

这篇关于SQL server 2005 UNPIVOT运算符的使用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/183456

相关文章

使用Java将DOCX文档解析为Markdown文档的代码实现

《使用Java将DOCX文档解析为Markdown文档的代码实现》在现代文档处理中,Markdown(MD)因其简洁的语法和良好的可读性,逐渐成为开发者、技术写作者和内容创作者的首选格式,然而,许多文... 目录引言1. 工具和库介绍2. 安装依赖库3. 使用Apache POI解析DOCX文档4. 将解析

Qt中QUndoView控件的具体使用

《Qt中QUndoView控件的具体使用》QUndoView是Qt框架中用于可视化显示QUndoStack内容的控件,本文主要介绍了Qt中QUndoView控件的具体使用,具有一定的参考价值,感兴趣的... 目录引言一、QUndoView 的用途二、工作原理三、 如何与 QUnDOStack 配合使用四、自

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

数据库面试必备之MySQL中的乐观锁与悲观锁

《数据库面试必备之MySQL中的乐观锁与悲观锁》:本文主要介绍数据库面试必备之MySQL中乐观锁与悲观锁的相关资料,乐观锁适用于读多写少的场景,通过版本号检查避免冲突,而悲观锁适用于写多读少且对数... 目录一、引言二、乐观锁(一)原理(二)应用场景(三)示例代码三、悲观锁(一)原理(二)应用场景(三)示例

使用Python构建一个Hexo博客发布工具

《使用Python构建一个Hexo博客发布工具》虽然Hexo的命令行工具非常强大,但对于日常的博客撰写和发布过程,我总觉得缺少一个直观的图形界面来简化操作,下面我们就来看看如何使用Python构建一个... 目录引言Hexo博客系统简介设计需求技术选择代码实现主框架界面设计核心功能实现1. 发布文章2. 加

SQL表间关联查询实例详解

《SQL表间关联查询实例详解》本文主要讲解SQL语句中常用的表间关联查询方式,包括:左连接(leftjoin)、右连接(rightjoin)、全连接(fulljoin)、内连接(innerjoin)、... 目录简介样例准备左外连接右外连接全外连接内连接交叉连接自然连接简介本文主要讲解SQL语句中常用的表

SQL server配置管理器找不到如何打开它

《SQLserver配置管理器找不到如何打开它》最近遇到了SQLserver配置管理器打不开的问题,尝试在开始菜单栏搜SQLServerManager无果,于是将自己找到的方法总结分享给大家,对SQ... 目录方法一:桌面图标进入方法二:运行窗口进入方法三:查找文件路径方法四:检查 SQL Server 安

MySQL 中的 LIMIT 语句及基本用法

《MySQL中的LIMIT语句及基本用法》LIMIT语句用于限制查询返回的行数,常用于分页查询或取部分数据,提高查询效率,:本文主要介绍MySQL中的LIMIT语句,需要的朋友可以参考下... 目录mysql 中的 LIMIT 语句1. LIMIT 语法2. LIMIT 基本用法(1) 获取前 N 行数据(

shell编程之函数与数组的使用详解

《shell编程之函数与数组的使用详解》:本文主要介绍shell编程之函数与数组的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录shell函数函数的用法俩个数求和系统资源监控并报警函数函数变量的作用范围函数的参数递归函数shell数组获取数组的长度读取某下的

使用Python开发一个带EPUB转换功能的Markdown编辑器

《使用Python开发一个带EPUB转换功能的Markdown编辑器》Markdown因其简单易用和强大的格式支持,成为了写作者、开发者及内容创作者的首选格式,本文将通过Python开发一个Markd... 目录应用概览代码结构与核心组件1. 初始化与布局 (__init__)2. 工具栏 (setup_t