SqlServerPIVOT函数快速实现⾏转列,UNPIVOT实现列转⾏
(转)
我们在写Sql语句的时候没经常会遇到将查询结果⾏转列,列转⾏的需求,拼接sql字符串,然后使⽤sp_executesql执⾏sql字符串是⽐较常规的⼀种做法。但是这样做实现起来⾮常复杂,⽽在SqlServer2005中我们有了PIVOT/UNPIVOT函数可以快速实现⾏转列和列转⾏的操作。
PIVOT函数,⾏转列
PIVOT函数的格式如下
PIVOT(<;聚合函数>([聚合列值]) FOR [⾏转列前的列名] IN([⾏转列后的列名1],[⾏转列后的列名2],[⾏转列后的列名3],.......[⾏转列后的列名N])) <;聚合函数>就是我们使⽤的SUM,COUNT,AVG等Sql聚合函数,也就是⾏转列后计算列的聚合⽅式。
[聚合列值]要进⾏聚合的列名
[⾏转列前的列名]这个就是需要将⾏转换为列的列名。
[⾏转列后的列名]这⾥需要声明将⾏的值转换为列后的列名,因为转换后的列名其实就是转换前⾏的值,所以上⾯格式中的[⾏转列后的列名1],[⾏转列后的列名2],[⾏转列后的列名3],......[⾏转列后的列名N]其实就是[⾏转列前的列名]每⼀⾏的值。
下⾯我们来看⼀个例⼦有⼀张表名为[ShoppingCart]有三列[Week],[TotalPrice],[GroupId],数据和表结构如下所⽰:
CREATE TABLE [dbo].[ShoppingCart](
[Week] [int] NOT NULL,
[TotalPrice] [decimal](18, 0) NOT NULL,
[GroupId] [int] NULL
) ON [PRIMARY]
GO
ALTER TABLE [dbo].[ShoppingCart] ADD  DEFAULT ((0)) FOR [TotalPrice]
GO
INSERT [dbo].[ShoppingCart] ([Week], [TotalPrice], [GroupId]) VALUES (1, CAST(10 AS Decimal(18, 0)), 1)
GO
INSERT [dbo].[ShoppingCart] ([Week], [TotalPrice], [GroupId]) VALUES (2, CAST(20 AS Decimal(18, 0)), 1)
GO
INSERT [dbo].[ShoppingCart] ([Week], [TotalPrice], [GroupId]) VALUES (3, CAST(30 AS Decimal(18, 0)), 1)
GO
INSERT [dbo].[ShoppingCart] ([Week], [TotalPrice], [GroupId]) VALUES (4, CAST(40 AS Decimal(18, 0)), 1)
GO
INSERT [dbo].[ShoppingCart] ([Week], [TotalPrice], [GroupId]) VALUES (5, CAST(50 AS Decimal(18, 0)), 1)
GO
INSERT [dbo].[ShoppingCart] ([Week], [TotalPrice], [GroupId]) VALUES (6, CAST(60 AS Decimal(18, 0)), 1)
GO
INSERT [dbo].[ShoppingCart] ([Week], [TotalPrice], [GroupId]) VALUES (7, CAST(70 AS Decimal(18, 0)), 1)
GO
现在我们是⽤PIVOT函数将列[WEEK]的⾏值转换为列,并使⽤聚合函数Count(TotalPrice)来统计每⼀个Week列在转换前有多少⾏数据,语句如下所⽰:
select *
from ShoppingCart as C
PIVOT(count(TotalPrice) FOR [Week] IN([1],[2],[3],[4],[5],[6],[7])) AS T
查询结果如下:
我们可以看到PIVOT函数成功地将[ShoppingCart]表列[Week]的⾏值转换为了七列,并且每⼀列统计转换前的⾏数为1,这符合我们的预期结果。那么根据我们前⾯定义的PIVOT函数转换格式,在本例中我们有如下公式对应值:
<;聚合函数>本例中为Count
[聚合列值]本例中为[TotalPrice],统计了⾏转列前的⾏数
[⾏转列前的列名]本例中为[Week]
[⾏转列后的列名]本例中为[1],[2],[3],[4],[5],[6],[7]七个列,因为⾏转列前[ShoppingCart]表的[Week]列有七个值1,2,3,4,5,6,7,所以这⾥声明转换后的列名也为七个,对应这七个值分别为[1],[2],[3],[4],[5],[6],[7],PIVOT函数会将[ShoppingCart]表中[Week]列的值分别和[1],[2],
[3],[4],[5],[6],[7]这七列的列名进⾏匹配,然后计算<;聚合函数>(本例中为count(TotalPrice))得出转换后的列值。
另外如果我们在[⾏转列后的列名]中只声明了部分值,那么PIVOT函数只会针对这些部分值做⾏转列,⽽那些没有被声明为列的⾏值会在⾏转列后被忽略掉。例如我们下⾯的语句声明了只对表ShoppingCart中[Week]列的1,2,3三个值做⾏转列,但是实际上表ShoppingCart中列[Week]有1,2,3,4,5,6,7这7个值,那么剩下的4到7就会被PIVOT函数忽略掉,如下所⽰:
select *
from ShoppingCart as C
PIVOT(count(TotalPrice) FOR [Week] IN([1],[2],[3])) AS T
我们可以看到查询结果中PIVOT函数只针对表ShoppingCart中列[Week]的1,2,3三个值做了⾏转列,⽽4到7被忽略了。
需要注意的是PIVOT函数的查询结果中多了⼀列GroupId,这是因为PIVOT函数只⽤到了[ShoppingCart]表中的列[Week]和[TotalPrice],[ShoppingCart]表中还有⼀列[GroupId],PIVOT函数没有⽤到,所以PIVOT函数默认将[ShoppingCart]表中没有⽤到的列当做了Group By来处理,⽤来作为⾏转列后每⼀⾏数据分⾏的依据,⼜由于列[GroupId]在[ShoppingCart]表中全为值1,所以最后PIVOT函数在⾏转列后只有⼀⾏[GroupId]为1的数据,如果我们将[ShoppingCart]表列[GroupId]的值从只有1变成有1和2两种值,如下所⽰:
然后再执⾏PIVOT查询:
select *
from ShoppingCart as C
PIVOT(count(TotalPrice) FOR [Week] IN([1],[2],[3],[4],[5],[6],[7])) AS T
会得到如下结果:
我们看到这⼀次我们⽤PIVOT函数做⾏转列后得到了两⾏值,可以看到转换后列[3]和[4]在[GroupId]为2的这⼀⾏上为1,这就是因为我们将[ShoppingCart]表中[Week]为3和4的两⾏改成了[GroupId]为2后,[GroupId]有了两个值1和2,所以PIVOT函数⾏转列后就有两⾏值。
知道了PIVOT函数的⽤法之后,我们来看看PIVOT函数的⼏种错误⽤法:
在PIVOT函数的使⽤中有⼀点需要注意,那就是[⾏转列后的列名]必须是[⾏转列前的列名]的值,PIVOT函数才能成功执⾏,⽐如如下所⽰如果我们将[⾏转列后的列名]声明了⼀个和[⾏转列前的列名]值毫不相⼲的数字1000,那么PIVOT函数执⾏后1000是没有任何数据的为0:
select *
from ShoppingCart as C
PIVOT(count(TotalPrice) FOR [Week] IN([1],[2],[3],[4],[5],[6],[7],[1000])) AS T
这是因为[ShoppingCart]表中列[Week]没有值1000,所以⽤PIVOT函数将列[Week]⾏转列后列[1000]的值就为0。
如果将PIVOT函数中[⾏转列后的列名]声明为了[⾏转列前的列名]完全不同的数据类型,还会导致PIVOT函数报错,例如下⾯我们在[⾏转列后的列名]中声明了⼀个列名为字符串[TestColumnName],但是由于[⾏转列前的列名]Week是Int类型,从⽽⽆法将字符串TestColumnName转换为Int类型,所以PIVOT函数报错了:
select *
from ShoppingCart as C
PIVOT(count(TotalPrice) FOR [Week] IN([1],[2],[3],[4],[5],[6],[7],[TestColumnName])) AS T
UNPIVOT函数,列转⾏
UNPIVOT函数的格式如下:
UNPIVOT([转换为⾏的列值在转换后对应的列名] for [转换为⾏的列名在转换后对应的列名] in ([转换为⾏的列1],[转换为⾏的列2],[转换为⾏的列3],...[转换为⾏的列N])) [转换为⾏的列值在转换后对应的列名]这个是进⾏列转⾏的列其数据值在转换为⾏后的列名称
[转换为⾏的列名在转换后对应的列名]这个是进⾏列转⾏的列其列名在转换为⾏后的列名称
[转换为⾏的列]这个是声明哪些列要进⾏列转⾏
如下所⽰,列转⾏前为:
select *
from ShoppingCart as C
PIVOT(count(TotalPrice) FOR [Week] IN([1],[2],[3],[4],[5],[6],[7])) AS T
现在使⽤UNPIVOT函数将上⾯结果的列[1],[2],[3],[4],[5],[6],[7]转换为⾏值,如下所⽰:
with PIVOT_Table as
(
select *
from ShoppingCart as C
PIVOT(count(TotalPrice) FOR [Week] IN([1],[2],[3],[4],[5],[6],[7])) AS T
)
select * from PIVOT_Table UNPIVOT([RowCount] for [Week] in ([1],[2],[3],[4],[5],[6],[7])) as T
可以看到[1],[2],[3],[4],[5],[6],[7]这七列在UNPIVOT函数执⾏后其值变为了列[RowCount],列转⾏前的列名称在转换后变为了列[Week],同样套⽤UNPIVOT函数格式我们可以得到如下结果:
[转换为⾏的列值在转换后对应的列名]在本例中为[RowCount]
tabletotal函数[转换为⾏的列名在转换后对应的列名]在本例中为[Week]
[转换为⾏的列]这个是声明哪些列要进⾏列转⾏,在本例中为[1],[2],[3],[4],[5],[6],[7]这七列
需要注意如果列转⾏前有两⾏值:
select *
from ShoppingCart as C
PIVOT(count(TotalPrice) FOR [Week] IN([1],[2],[3],[4],[5],[6],[7])) AS T
那么UNPIVOT函数转换后应该为14⾏(列转⾏前的⾏数2 X 需要进⾏列转⾏的列数7 = 14)数据:
with PIVOT_Table as
(
select *
from ShoppingCart as C
PIVOT(count(TotalPrice) FOR [Week] IN([1],[2],[3],[4],[5],[6],[7])) AS T
)
select * from PIVOT_Table UNPIVOT([RowCount] for [Week] in ([1],[2],[3],[4],[5],[6],[7])) as T
此外需要注意UNPIVOT函数不会对列转⾏中没有⽤到的列作Group By处理,也不会对列传⾏后的值做聚合运算,这⼀点是和PIVOT函数不同的。⽐如现在如果我们有下⾯⼀个查询:
with PIVOT_Table as
(
select *
from ShoppingCart as C
PIVOT(count(TotalPrice) FOR [Week] IN([1],[2],[3],[4],[5],[6],[7])) AS T
union all
select *
from ShoppingCart as C
PIVOT(count(TotalPrice) FOR [Week] IN([1],[2],[3],[4],[5],[6],[7])) AS T
)
select * from PIVOT_Table
起查询结果为:
我们可以看到查询结果中有两⾏GroupId为1的数据,现在我们再⽤UNPIVOT函数对这个查询的列[1]到[7]做列转⾏运算,其中没有⽤到列GroupId:
with PIVOT_Table as
(
select *
from ShoppingCart as C
PIVOT(count(TotalPrice) FOR [Week] IN([1],[2],[3],[4],[5],[6],[7])) AS T
union all
select *
from ShoppingCart as C
PIVOT(count(TotalPrice) FOR [Week] IN([1],[2],[3],[4],[5],[6],[7])) AS T
)
select * from PIVOT_Table UNPIVOT([RowCount] for [Week] in ([1],[2],[3],[4],[5],[6],[7])) as T
结果如下所⽰:
我们可以看到结果出现了14⾏数据(列转⾏前的⾏数2 X 需要进⾏列转⾏的列数7 = 14),所以我们可
以看到虽然我们在UNPIVOT函数中没有⽤到列GroupId,并且在列转⾏前GroupId列有两⾏相同的值1,但是UNPIVOT函数在列转⾏后仍然⽣成了14⾏数据,⽽不是7⾏数据,因此并没有对GroupId列做Group By处理来合并相同的值,这⼀点和前⾯的PIVOT函数是不同的。
转载来源

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。