当前位置:  开发笔记 > 数据库 > 正文

SQL Server 2005透视未知列数

如何解决《SQLServer2005透视未知列数》经验,为你挑选了2个好方法。

我正在处理一组类似于以下内容的数据.

StudentName  | AssignmentName |  Grade
---------------------------------------
StudentA     | Assignment 1   | 100
StudentA     | Assignment 2   | 80
StudentA     | Total          | 180
StudentB     | Assignment 1   | 100
StudentB     | Assignment 2   | 80
StudentB     | Assignment 3   | 100
StudentB     | Total          | 280

作业的名称和数量是动态的,我需要将结果与以下内容相似.

Student      | Assignment 1  | Assignment 2  | Assignment 3  | Total
--------------------------------------------------------------------
Student A    | 100           | 80            | null          | 180
Student B    | 100           | 80            | 100           | 280

理想情况下,我希望根据可能包含/关联每个分配的"截止日期"对列进行排序.如果可能,总数应该在最后(如果可能,可以计算并从查询中删除.)

我知道如何通过简单地命名列来使用数据透视表进行3次分配,它试图以动态的方式进行,我还没有找到好的解决方案.我试图在SQL Server 2005上执行此操作

编辑

理想情况下,我想在不使用动态SQL的情况下实现此功能,因为这违反了策略.如果不可能......那么使用Dynamic SQL的工作示例将起作用.



1> Cade Roux..:

我知道你说没有动态SQL,但我认为没有办法直接做到这一点SQL.

如果你在数据透视表和连接列和sql 2005中的PIVOT中查看我对类似问题的答案

动态SQL不容易注射,没有充分理由禁止注射.另一种可能性(如果数据很少变化)是进行代码生成 - 而不是动态的SQL,SQL定期生成存储过程.



2> Taryn..:

对于PIVOT使用动态sql的数据,您可以在SQL Server 2005+中使用以下代码:

创建表:

CREATE TABLE yourtable
    ([StudentName] varchar(8), [AssignmentName] varchar(12), [Grade] int)
;

INSERT INTO yourtable
    ([StudentName], [AssignmentName], [Grade])
VALUES
    ('StudentA', 'Assignment 1', 100),
    ('StudentA', 'Assignment 2', 80),
    ('StudentA', 'Total', 180),
    ('StudentB', 'Assignment 1', 100),
    ('StudentB', 'Assignment 2', 80),
    ('StudentB', 'Assignment 3', 100),
    ('StudentB', 'Total', 280)
;

动态PIVOT:

DECLARE @cols AS NVARCHAR(MAX),
    @query  AS NVARCHAR(MAX)

select @cols = STUFF((SELECT distinct ',' + QUOTENAME(AssignmentName) 
                    from yourtable
            FOR XML PATH(''), TYPE
            ).value('.', 'NVARCHAR(MAX)') 
        ,1,1,'')

set @query = 'SELECT StudentName, ' + @cols + ' from 
             (
                select StudentName, AssignmentName, grade
                from yourtable
            ) x
            pivot 
            (
                min(grade)
                for assignmentname in (' + @cols + ')
            ) p '

execute(@query)

请参阅SQL Fiddle with Demo

结果是:

| STUDENTNAME | ASSIGNMENT 1 | ASSIGNMENT 2 | ASSIGNMENT 3 | TOTAL |
--------------------------------------------------------------------
|    StudentA |          100 |           80 |       (null) |   180 |
|    StudentB |          100 |           80 |          100 |   280 |

推荐阅读
黄晓敏3023
这个屌丝很懒,什么也没留下!
DevBox开发工具箱 | 专业的在线开发工具网站    京公网安备 11010802040832号  |  京ICP备19059560号-6
Copyright © 1998 - 2020 DevBox.CN. All Rights Reserved devBox.cn 开发工具箱 版权所有