T-SQL语句中,Pivot运算符用于在列和行之间对数据进行旋转或透视转换,PIVOT命令可以实现数据表的列转行,同时执行聚合运算,UNPIVOT则与其相反,实现数据的行转列。
PIVOT通过将表达式某一列中的唯一值转换为输出中的多个列来旋转表值表达式,并在必要时对最终输出中所需的任何其余列值执行聚合。UNPIVOT与PIVOT执行相反的操作,将表值表达式的列转换为列值。
通俗简单的说:PIVOT就是行转列,UNPIVOT就是列传行

一.PIVOT(行转列)

1. 建表

建立一个销售情况表,其中,year字段表示年份,quarter字段表示季度,amount字段表示销售额。quarter字段分别用Q1, Q2, Q3, Q4表示一、二、三、四季度。

  1. --01.建表
  2. Create table SalesByQuarter
  3. (
  4. year INT,
  5. quarter CHAR(2),
  6. amount MONEY
  7. )

2. 填入表数据

使用如下程序填入表数据。

--02.填入表数据
SET NOCOUNT ON 
DECLARE @index INT
DECLARE @q INT
SET @index=0
DECLARE @year INT 
WHILE(@index <100)
    BEGIN
        SET @year=2005+(@index % 4)
        SET @q=(CAST((RAND()*500) AS INT) % 4)+1
        INSERT INTO SalesByQuarter VALUES(@year,'Q'+CAST(@q AS CHAR(1)),RAND()*10000.00)
        SET @index=@index+1
    END

3. 如果我们要比较每年中各季度的销售状况,要怎么办呢?有以下两种方法:

(1)使用传统Select的CASE语句查询

在SQL Server以前的版本里,将行级数据转换为列级数据就要用到一系列CASE语句和聚合查询。虽然这种方式让开发人员具有了对所返回数据进行高度控制的能力,但是编写出这些查询是一件很麻烦的事情。

SELECT year as 年份,
    sum(case when quarter='Q1' then amount else 0 end) 第一季度,
    sum(case when quarter='Q2' then amount else 0 end) 第二季度,
    sum(case when quarter='Q3' then amount else 0 end) 第三季度,
    sum(case when quarter='Q4' then amount else 0 end) 第四季度
FROM SalesByQuarter
GROUP BY year
ORDER BY year DESC

原始表Old
图片.png
得到的结果New如下:
图片.png

(2)使用PIVOT

由于SQL Server 2005有了新的PIVOT运算符,就不再需要CASE语句和GROUP BY语句了。(每个PIVOT查询都涉及某种类型的聚合,因此你可以忽略GROUP BY语句。)PIVOT运算符让我们能够利用CASE语句查询实现相同的功能,但是你可以用更少的代码就实现,而且看起来更漂亮。

SElECT 
  year as 年份,
    Q1 AS 第一季度,
    Q2 AS 第二季度,
    Q3 AS 第三季度,
    Q4 AS 第四季度
FROM SalesByQuarter PIVOT(SUM(amount)FOR quarter IN(Q1,Q2,Q3,Q4)) AS P 
ORDER BY YEAR DESC

得到的结果如下:
图片.png

二 UNPIVOT

1 建表

建立一个通讯录表

CREATE TABLE PhoneNumbers    
(
    PersonID int,
    HomePhone varchar(12),
    CellPhone varchar(12),
    WorkPhone varchar(12),
    FaxPhone varchar(12),
)

2. 填入表数据

使用如下程序填入表数据。

INSERT INTO PhoneNumbers
VALUES(1,'0527-836-01','188-123-01','0527','FAX01'),
      (2,'0527-836-02','188-123-02','0527','FAX02'),
      (3,'0527-836-03','188-123-03','0527','FAX03'),
      (4,'0527-836-04','188-123-04','0527','FAX04')

3. 使用PIVOT列转行

SELECT PersonID,PhoneType,PhoneNumber
FROM 
(SELECT PersonID,HomePhone,CellPhone,WorkPhone,FaxPhone FROM PhoneNumbers) AS SRC
    UNPIVOT
    (PhoneNumber FOR PhoneType IN (HomePhone,CellPhone,WorkPhone,FaxPhone)) AS UNPVT

得到的结果如下:
Old
图片.png
New
图片.png