在SQL Server 2005中实现表的行列转换

2016-02-19 17:44 1 1 收藏

关注图老师设计创意栏目可以让大家能更好的了解电脑,知道有关于电脑的更多有趣教程,今天给大家分享在SQL Server 2005中实现表的行列转换教程,希望对大家能有一点小小的帮助。

【 tulaoshi.com - 编程语言 】

  PIVOT和UNPIVOT关系运算符是SQL Server 2005提供的新增功能,因此,对升级到SQL Server 2005的数据库使用PIVOT和UNPIVOT时,数据库的兼容级别必须设置为90(可以使用sp_dbcmptlevel存储过程设置兼容级别)。

  在查询的FROM子句中使用PIVOT和UNPIVOT,可以对一个输入表值表达式执行某种操作,以获得另一种形式的表。PIVOT运算符将输入表的行旋转为列,并能同时对行执行聚合运算。而UNPIVOT运算符则执行与PIVOT运算符相反的操作,它将输入表的列旋转为行。

  在FROM子句中使用PIVOT和UNPIVOT关系运算符时的语法格式如下:

  

[ FROM { <table_source> } [ ,...n ] ]<table_source> ::= { table_or_view_name [ [ AS ] table_alias ] <pivoted_table> | <unpivoted_table>}<pivoted_table> ::=table_source PIVOT <pivot_clause> table_alias<pivot_clause> ::=( aggregate_function ( value_column ) FOR pivot_column  IN ( <column_list> ))<unpivoted_table> ::=table_source UNPIVOT <unpivot_clause> table_alias<unpivot_clause> ::=( value_column FOR pivot_column IN ( <column_list> ) )<column_list> ::= column_name [ , ... ] table_source PIVOT <pivot_clause>

  指定对table_source表中的pivot_column列进行透视。table_source可以是一个表、表表达式或子查询。

  aggregate_function

  系统或用户定义的聚合函数。注意:不允许使用COUNT(*)系统聚合函数。

  value_column

  PIVOT运算符用于进行计算的值列。与UNPIVOT一起使用时,value_column不能是输入table_source中的现有列的名称。

(本文来源于图老师网站,更多请访问http://www.tulaoshi.com/bianchengyuyan/)

  FOR pivot_column

  PIVOT运算符的透视列。pivot_column必须是可隐式或显式转换为nvarchar()的类型。

  使用UNPIVOT时,pivot_column是从table_source中提取输出的列名称,table_source中不能有该名称的现有列。

(本文来源于图老师网站,更多请访问http://www.tulaoshi.com/bianchengyuyan/)

  IN ( column_list )

  在PIVOT子句中,column_list列出pivot_column中将成为输出表的列名的值。

  在UNPIVOT子句中,column_list列出table_source中将被提取到单个pivot_column中的所有列名。

  table_alias

  输出表的别名。

  UNPIVOT < unpivot_clause >

  指定将输入表中由column_list指定的多个列的值缩减为名为pivot_column的单个列。

  常见的可能会用到PIVOT的情形是:需要生成交叉表格报表以汇总数据。交叉表是使用较为广泛的一种表格式,例如,图5-4所示的产品销售表就是一个典型的交叉表,其中的月份和产品种类都可以继续添加。但是,这种格式在进行数据表存储的时候却并不容易管理,要存储图5-4这样的表格数据,数据表通常需要设计为图5-5这样的结构。这样就带来一个问题,用户既希望数据容易管理,又希望能够生成一种能够容易阅读的表格数据。好在PIVOT为这种转换提供了便利。

  图5-4 产品销售表 图5-5 数据表结构

  假设Sales.Orders表中包含有ProductID(产品ID)、OrderMonth(销售月份)和SubTotal(销售额)列,并存储有如表5-2所示的内容。

  表5-2 Sales.Orders表中的内容

ProductID OrderMonth SubTotal 15 100.00 1 6 100.00 2 5 200.00 26 200.00 2 7 300.00 35400.00 35400.00

  执行下面的语句:

  

SELECT ProductID, [5] AS 五月, [6] AS 六月, [7] AS 七月FROMSales.Orders PIVOT(SUM (Orders.SubTotal)FOR Orders.OrderMonth IN( [5], [6], [7] )) AS pvtORDER BY ProductID;

  在上面的语句中,Sales.Orders是输入表,Orders.OrderMonth是透视列(pivot_column),Orders.SubTotal是值列(value_column)。上面的语句将按下面的步骤获得输出结果集:

  a.PIVOT首先按值列之外的列(ProductID和OrderMonth)对输入表Sales.Orders进行分组汇总,类似执行下面的语句:

  

SELECT ProductID,OrderMonth,SUM (Orders.SubTotal) AS SumSubTotalFROM Sales.OrdersGROUP BY ProductID,OrderMonth;

  这时候将得到一个如表5-3所示的中间结果集。其中只有ProductID为3的产品由于在5月有2笔销售记录,被累加到了一起(值为800)。

  表5-3 Sales.Orders表经分组汇总后的结果

ProductIDOrderMonthSumSubTotal 1 5 100.00 1 6 100.00 2 5 200.00 2 6 200.00 27300.00 35800.00

  b.PIVOT根据FOR Orders.OrderMonth IN指定的值5、6、7,首先在结果集中建立名为5、6、7的列,然后从图5-3所示的中间结果中取出OrderMonth列中取出相符合的值,分别放置到5、6、7的列中。此时得到的结果集的别名为pvt(见语句中AS pvt的指定)。结果集的内容如表5-4所示。

  表5-4 使用FOR Orders.OrderMonth IN( [5], [6], [7] )后得到的结果集

ProductID 5 6 71 100.00 100.00 NULL2 200.00 200.00200.003 800.00NULLNULL

  c.最后根据SELECT ProductID, [5] AS 五月, [6] AS 六月, [7] AS 七月FROM的指定,从别名pvt结果集中检索数据,并分别将名为5、6、7的列在最终结果集中重新命名为五月、六月、七月。这里需要注意的是FROM的含义,其表示从经PIVOT关系运算符得到的pvt结果集中检索数据,而不是从Sales.Orders中检索数据。最终得到的结果集如表5-5所示。

  表5-5 由表5-2所示的Sales.Orders表将行转换为列得到的最终结果集

ProductID五月六月七月 1 100.00 100.00 NULL 2 200.00200.00200.00 3 800.00 NULL NULL

  UNPIVOT与PIVOT执行几乎完全相反的操作,将列转换为行。但是,UNPIVOT并不完全是PIVOT的逆操作,由于在执行PIVOT过程中,数据已经被进行了分组汇总,所以使用UNPIVOT并不会重现原始表值表达式的结果。假设表5-5所示的结果集存储在一个名为MyPvt的表中,现在需要将列标识符五月、六月和七月转换到对应于相应产品ID的行值(即返回到表5-3所示的格式)。这意味着必须另外标识两个列,一个用于存储月份,一个用于存储销售额。为了便于理解,仍旧分别将这两个列命名为OrderMonth和SumSubTotal。参考下面的语句:

  

CREATE TABLE MyPvt (ProductID int, 五月int, 六月 int, 七月int); --建立MyPvt表GO  --将表5-5中所示的值插入到MyPvt表中INSERT INTO MyPvt VALUES (1,100,100,0);INSERT INTO MyPvt VALUES (2,200,200,200);INSERT INTO MyPvt VALUES (3,800,0,0);  --执行UNPIVOTSELECT ProductID, OrderMonth, SubTotalFROM MyPvt UNPIVOT (SubTotal FOR OrderMonth IN  (五月, 六月, 七月) )AS unpvt;

来源:http://www.tulaoshi.com/n/20160219/1615930.html

延伸阅读
问题提出 在我的新工作中,我面临着一个问题,在product数据库中把创建的表进行归档,因为随着数据库的增长,现在需要把它们移走。同时,这些归档的表被几项工作使用,而且它还在应用程序的代码中被调用。这样移动它们就是一个非常复杂的过程。我希望能找到一种方法,尽可能减轻开发人员必须得完成的工作量,因为他们的时间非常有限...
标签: SQLServer
分页,就是按照某种规则显示分组数据集,但是在SQL Server 中,分页并不是十分容易就能够实现。在过去,开发人员通常需要自己编写程序,使用临时表格来实现分页功能,或者将所有的数据结果集返回到客户端,在客户端进行分页操作。从开发人员或者DBA的角度来看,两种方法都不能令人满意。 随着SQL Server的发布,其中的一些排序函数使得开发人员...
老南瓜一直对SQL Server 2005里的用户的概念不是很清楚,如果你和我当初一样,认为下面的两个用户是一个概念,那就有必要点上一根烟,喝上一品茶,细细品味一下这之间的不同了。 左边的图标出了一个可以访问AdventureWorks数据库的用户,而右边的图则标出了整个数据库服务器所有的“用户”。 大家知道,SQL Server 2005有两种...
标签: Web开发
通过ADO可以访问SQL SERVER,并执行相应的SQL语句建库、建表,下面是SQL SERVER BOOKS ONLINE中的相关定义。     建表:   CREATE TABLE   [       database_name.[owner].       | owner. &n...
标签: SQLServer
SQL Server 2005中的T-SQL增强(一) 丰富的数据类型 Richer Data Types 1、varchar(max)、nvarchar(max)和varbinary(max)数据类型最多可以保存2GB的数据,可以取代text、ntext或image数据类型。 CREATE TABLE myTable(id INT,content VARCHAR(MAX)) 2、XML数据类型 XML数据类型允许用户在SQL Server数据库中保存XML片...

经验教程

622

收藏

56
微博分享 QQ分享 QQ空间 手机页面 收藏网站 回到头部