Pivot di SQL Server 2005 su un numero sconosciuto di colonne
-
03-07-2019 - |
Domanda
Sto lavorando con un set di dati simile al seguente.
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
Il nome e il numero di incarichi sono dinamici, devo ottenere risultati simili ai seguenti.
Student | Assignment 1 | Assignment 2 | Assignment 3 | Total -------------------------------------------------------------------- Student A | 100 | 80 | null | 180 Student B | 100 | 80 | 100 | 280
Ora idealmente vorrei ordinare la colonna in base a una "data di scadenza". che potrebbero essere inclusi / associati a ciascun incarico. Il totale dovrebbe essere alla fine, se possibile (può essere calcolato e rimosso dalla query, se possibile.)
So come farlo per i 3 compiti usando pivot semplicemente nominando le colonne, sta provando a farlo in modo dinamico che non ho ancora trovato una BUONA soluzione. Sto provando a farlo su SQL Server 2005
Modifica
Idealmente, vorrei implementare questo SENZA utilizzando Dynamic SQL, poiché ciò è contrario alla politica. Se non è possibile ... funzionerà un esempio funzionante con SQL dinamico.
Soluzione
So che non hai detto SQL
dinamico, ma non vedo alcun modo di farlo in SQL
.
Se dai un'occhiata alle mie risposte a problemi simili su Tabella pivot e Concatenare colonne e PIVOT in sql 2005
Il SQL
dinamico non è vulnerabile all'iniezione e non c'è motivo di proibirlo. Un'altra possibilità (se i dati cambiano molto raramente) è quella di generare codice - invece che SQL
dinamico, il SQL
viene generato su una procedura memorizzata su base regolare.
Altri suggerimenti
Per PIVOT
questi dati utilizzando sql dinamico è possibile utilizzare il seguente codice in SQL Server 2005+:
Crea tabella:
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 dinamico:
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)
Vedi SQL Fiddle with Demo
Il risultato è:
| STUDENTNAME | ASSIGNMENT 1 | ASSIGNMENT 2 | ASSIGNMENT 3 | TOTAL |
--------------------------------------------------------------------
| StudentA | 100 | 80 | (null) | 180 |
| StudentB | 100 | 80 | 100 | 280 |
L'unico modo in cui ho scoperto di farlo è utilizzare SQL dinamico e inserire le etichette delle colonne in una variabile.
è possibile eseguire una query di information_schema per ottenere i nomi e i tipi di colonna, quindi utilizzare il risultato come sottoquery durante la creazione del set di risultati. Nota che probabilmente dovrai modificare un po 'l'accesso di accesso.
È lo stesso di PIVOT in sql 2005
Se questi dati sono destinati al consumo in un report, è possibile utilizzare una matrice SSRS. Genererà colonne in modo dinamico dal set di risultati. L'ho usato molte volte: funziona abbastanza bene per i report a campi incrociati dinamici.
Ecco un buon esempio con sql dinamico. http: //www.simple- talk.com/community/blogs/andras/archive/2007/09/14/37265.aspx
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
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