SQLServer2005 Pivot 转置使用动态列(应用到视图)

SQLServer2005 Pivot 转置使用动态列(应用到视图)

最近项目中用到Pivot 对表进行转置,遇到一些问题,主要是Pivot 转置的时候没有办法动态产生转置列名,而作视图的时候又很需要动态的产生这些列,百度上似乎也没有找的很满意的答案,在google上搜到一老外的解决方案,现在自己总结了一下,希望给用的上的朋友一些帮助。

images/pivot20080719001.gif

 

 

1.创建表脚本

 

if exists (select 1

            from  sysobjects

           where  id = object_id('Insurances')

            and   type = 'U')

   drop table Insurances

go

 

/*==============================================================*/

/* Table: Insurances                                            */

/*==============================================================*/

create table Insurances (

   RefID                uniqueidentifier     not null,

   HRMS                 nvarchar(20)         null,

   Name                 nvarchar(20)         null,

   InsuranceMoney       money                null,

   InsuranceName        nvarchar(100)        not null,

   constraint PK_INSURANCES primary key (RefID)

)

go

 

 

2.测试数据脚本

 

insert into Insurances values (newid(),1,'张三',200,'养老保险')

insert into Insurances values (newid(),1,'张三',300,'医疗保险')

insert into Insurances values (newid(),2,'李四',250,'养老保险')

insert into Insurances values (newid(),2,'李四',350,'医疗保险')

insert into Insurances values (newid(),3,'王二',150,'养老保险')

insert into Insurances values (newid(),3,'王二',300,'医疗保险')

 

3.查询表数据

 

select HRMS,Name,InsuranceMoney,InsuranceName From Insurances

 

HRMS                 Name                 InsuranceMoney        InsuranceName

-------------------- -------------------- --------------------- ----------

1                    张三                  200.00                养老保险

2                    李四                  350.00                医疗保险

2                    李四                  250.00                养老保险

1                    张三                  300.00                医疗保险

3                    王二                  300.00                医疗保险

3                    王二                  150.00                养老保险

 

4.转置表数据

 

select * from

(

select HRMS,Name,InsuranceMoney,InsuranceName from Insurances

) p

Pivot (

sum(InsuranceMoney)

FOR InsuranceName IN

( [医疗保险], [养老保险]))

as pvt

 

 

HRMS                 Name                 医疗保险                 养老保险

-------------------- -------------------- --------------------- ---------------------

2                    李四                  350.00                250.00

3                    王二                  300.00                150.00

1                    张三                  300.00                200.00

 

5.偶的问题

 

 images/pivot20080719002.gif

 

 

 

这个语句中 医疗保险、养老保险 SQL语句中写死的,而且Sql2005中这个代码没有办法使用动态的查询结果集

 

5.存储过程解决问题

 

所以如果要动态的完成个脚本,可以先拼出SQL 然后通过exec sp_executesql 执行

 

实现存储过程

 

create procedure InsurancePivot

as

Begin

    DECLARE @ColumnNames VARCHAR(3000)

 

    SET @ColumnNames=''

 

    SELECT

       @ColumnNames = @ColumnNames + '[' + InsuranceName + '],'

    FROM

       (

       SELECT DISTINCT InsuranceName FROM Insurances

       ) t

 

    SET @ColumnNames= LEFT(@ColumnNames, LEN(@ColumnNames)-1)

 

    DECLARE @selectSQL NVARCHAR(3000)

 

    SET @selectSQL=

    'SELECT HRMS,Name,{0} FROM

       (

       SELECT HRMS,Name,InsuranceMoney,InsuranceName FROM Insurances

       ) p

     Pivot( Max(InsuranceMoney)  For InsuranceName in ({0})) AS pvt

       ORDER BY HRMS'

 

    SET @selectSQL= REPLACE(@selectSQL,'{0}',@ColumnNames)

 

    exec sp_executesql @selectSQL

end

 

测试存储过程:

 

exec InsurancePivot

 

HRMS                 Name                 养老保险                 医疗保险

-------------------- -------------------- --------------------- ---------------------

1                    张三                  200.00                300.00

2                    李四                  250.00                350.00

3                    王二                  150.00                300.00

 

 

6.关于视图的新问题和解决方案

 

在视图中没有办法直接调用这个存储过程,但是我们在做程序、做报表的时候又非常需要

 

其实可以通过OPENQUERY来实现(这是一个非正规的解决方式,但目前可以实现)

(另外可以使用OPENROWSET,但是参数太多偶放弃了)

 

使用OPENQUERY 的格式是:OPENQUERY([链接服务器],’sql语句’)

 

因为是当前数据的视图, 链接服务器可以通过属性查看,MSCBF107 是我测试的链接服务器

 

images/pivot20080719003.gif

 

images/pivot20080719004.gif

 

 

也可以通过sp_helpserver 查看

 

images/pivot20080719006.gif

 

 

下面这句话也非常重要,使用的朋友替换[MSCBF107]ok了,否则使用OPENQUERY会出现未将服务器'MSCBF107' 配置为用于DATA ACCESS

 

sp_serveroption [MSCBF107], 'Data Access', 'True'

 

创建视图如下:

 

 

create view InsurancePivotView

as

select *From OPENQUERY ([MSCBF107],N'SET FMTONLY OFF;exec test.dbo.InsurancePivot')

 

 

测试视图就可以得到想要的结果了

 

select *from InsurancePivotView

That’s all
[附件下载]
cbf1072022/10/17 11:00:53
看起来像​2012 创建临时表的错误, 这里的语法应该是有变化了
kk2022/10/13 16:52:59
消息 11514,级别 16,状态 1,过程 sp_describe_first_result_set,行 1 [批起始行 0]
无法确定元数据,因为过程“InsurancePivot”中的语句“exec sp_executesql @selectSQL”包含动态 SQL。请考虑使用 WITH RESULT SETS 子句显式描述结果集。


请问提示这个错误如何解决
ada2020/5/26 16:57:32
哈哈哈,很清晰,解决了一直没搞懂的问题,大谢啦
cbf1072018/7/16 11:25:47
zoe 业务层用代码去解决
zoe2018/6/13 9:35:54
动态的sql语句还有什么方式加啊
cbf1072017/3/27 10:01:47
呵呵,其实在业务层实现才是最终方案^_^
mike2017/3/15 13:18:36
漂亮  我就知道不止我一个人遇到这个问题。
mingming2016/6/4 12:36:04
​SQL SERVER 2012有错误
消息 11514,级别 16,状态 1,过程 sp_describe_first_result_set,第 1 行
无法确定元数据,因为过程“InsurancePivot”中的语句“exec sp_executesql @selectSQL”包含动态 SQL。请考虑使用 WITH RESULT SETS 子句显式描述结果集。
游客2014/9/23 14:40:26
很好,很感谢
qqqqqqq2014/2/19 14:23:34

很好

vt2014/1/10 10:49:23
ok,3ks
访客: