23

我正在处理一组如下所示的数据。

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

现在理想情况下,我想根据可以包含/关联到每个作业的“到期日期”对列进行排序。如果可能,总数应该放在最后(如果可能,可以计算并从查询中删除。)

我知道如何使用 pivot 简单地命名列来完成 3 个作业,它试图以我还没有找到好的解决方案的动态方式来做。我正在尝试在 SQL Server 2005 上执行此操作

编辑

理想情况下,我想在不使用动态 SQL 的情况下实现这一点,因为这违反了政策。如果不可能……那么使用动态 SQL 的工作示例将起作用。

4

7 回答 7

12

我知道你说没有动态SQL,但我看不出有什么办法可以直接做到这一点SQL

如果您在 sql 2005 中的Pivot Table and Concatenate ColumnsPIVOT查看我对类似问题的回答

那里的动态SQL不容易被注入,也没有充分的理由禁止它。另一种可能性(如果数据很少更改)是进行代码生成 - 而不是动态SQLSQL定期生成到存储过程。

于 2008-10-17T21:05:10.877 回答
10

对于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)
;

动态枢轴:

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

结果是:

| STUDENTNAME | ASSIGNMENT 1 | ASSIGNMENT 2 | ASSIGNMENT 3 | TOTAL |
--------------------------------------------------------------------
|    StudentA |          100 |           80 |       (null) |   180 |
|    StudentB |          100 |           80 |          100 |   280 |
于 2012-12-21T14:33:10.840 回答
1

我发现这样做的唯一方法是使用动态 SQL 并将列标签放入变量中。

于 2008-10-17T20:24:38.587 回答
0

您可以查询 information_schema 以获取列名和类型,然后在构建结果集时将结果用作子查询。请注意,您可能需要稍微更改登录的访问权限。

于 2008-10-17T20:24:59.507 回答
0

这与sql 2005 中的 PIVOT相同

如果此数据用于报告中的消费,您可以使用 SSRS 矩阵。它将从结果集中动态生成列。我已经用过很多次了——它对于动态交叉表报告非常有效。

这是一个带有动态 sql 的好例子。 http://www.simple-talk.com/community/blogs/andras/archive/2007/09/14/37265.aspx

于 2008-10-17T23:23:19.537 回答
-1
SELECT TrnType
INTO #Temp1
FROM
(
    SELECT '[' + CAST(TransactionType AS VARCHAR(4)) + ']' AS TrnType FROM tblPaymentTransactionTypes
) AS tbl1

SELECT * FROM #Temp1

SELECT * FROM
(
    SELECT FirstName + ' ' + LastName AS Patient, TransactionType, ISNULL(PostedAmount, 0) AS PostedAmount
    FROM tblPaymentTransactions
            INNER JOIN emr_PatientDetails ON tblPaymentTransactions.PracticeID = emr_PatientDetails.PracticeId
            INNER JOIN tblPaymentTransactionDetails ON emr_PatientDetails.PatientId = tblPaymentTransactionDetails.PatientID
                        AND tblPaymentTransactions.TransactionID = tblPaymentTransactionDetails.TransactionID
    WHERE emr_PatientDetails.PracticeID = 152
) tbl
PIVOT (SUM(PostedAmount) FOR [TransactionType] IN (SELECT * FROM #Temp1)
) AS tbl4
于 2012-09-25T11:07:25.777 回答
-3
select studentname,[Assign1],[Assign2],[Assign3],[Total] 
from 
(
 select studentname, assignname, grade from student
)s
pivot(sum(Grade) for assignname IN([Assign1],[Assign2],[Assign3],[Total])) as pvt
于 2010-02-05T13:55:31.060 回答