Сводная таблица SQL Server 2005 для неизвестного числа столбцов - PullRequest
19 голосов
/ 18 октября 2008

Я работаю с набором данных, который выглядит примерно так:

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-х назначений, используя pivot с простым именованием столбцов, он пытается сделать это динамическим образом, для которого я пока не нашел ХОРОШЕГО решения. Я пытаюсь сделать это на SQL Server 2005

EDIT

В идеале я хотел бы реализовать это БЕЗ использования динамического SQL, поскольку это противоречит политике. Если это невозможно ... тогда будет работать рабочий пример с динамическим SQL.

Ответы [ 7 ]

12 голосов
/ 18 октября 2008

Я знаю, что вы сказали, что нет динамики SQL, но я не вижу способа сделать это прямо SQL.

Если вы посмотрите мои ответы на похожие проблемы в Сводная таблица и объединенные столбцы и PIVOT в sql 2005

Динамический SQL там не уязвим для инъекций, и нет веских причин для его запрета. Другая возможность (если данные меняются очень редко) - генерировать код - вместо динамического SQL, SQL генерируется для хранимой процедуры на регулярной основе.

8 голосов
/ 21 декабря 2012

К 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 с демоверсией

Результат:

| STUDENTNAME | ASSIGNMENT 1 | ASSIGNMENT 2 | ASSIGNMENT 3 | TOTAL |
--------------------------------------------------------------------
|    StudentA |          100 |           80 |       (null) |   180 |
|    StudentB |          100 |           80 |          100 |   280 |
1 голос
/ 18 октября 2008

Единственный способ сделать это - использовать динамический SQL и поместить метки столбцов в переменную.

0 голосов
/ 25 сентября 2012
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
0 голосов
/ 05 февраля 2010
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
0 голосов
/ 18 октября 2008

Это то же самое, что и PIVOT в sql 2005

Если эти данные предназначены для использования в отчете, вы можете использовать матрицу SSRS. Он будет генерировать столбцы динамически из набора результатов. Я использовал его много раз - он отлично работает для динамических отчетов с перекрестными таблицами.

Вот хороший пример с динамическим sql. http://www.simple -talk.com / сообщества / блоги / Андраш / Архив / 2007/09/14 / 37265.aspx

0 голосов
/ 18 октября 2008

вы можете запросить информационную_схему, чтобы получить имена и типы столбцов, а затем использовать результат в качестве подзапроса при построении набора результатов. Обратите внимание, что вам, вероятно, придется немного изменить доступ для входа в систему.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...