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

理想的には、各割り当てに含める/関連付けることができる「期日」に基づいて列を並べ替えたいと思います。合計は、可能であれば最後にある必要があります (可能であれば、計算してクエリから削除できます)。

列に名前を付けるだけでピボットを使用して3つの割り当てを行う方法を知っています.動的な方法でそれを実行しようとしていますが、まだ良い解決策が見つかりません. 私はSQL Server 2005でこれをやろうとしています

編集

理想的には、ポリシーに反するため、動的 SQL を使用せずにこれを実装したいと考えています。それが不可能な場合は、動的 SQL を使用した実際の例が機能します。

4

7 に答える 7

12

あなたが no dynamic と言ったことは知っていますSQLが、それをまっすぐに行う方法がわかりませんSQL

Pivot Table and Concatenate Columns and PIVOT in sql 2005で同様の問題に対する私の回答をチェックしてください。

そこのダイナミクスSQLはインジェクションに対して脆弱ではなく、インジェクションを禁止する正当な理由はありません。別の可能性 (データが頻繁に変更されない場合) は、コード生成を行うことです。動的SQLではなくSQL、定期的にストアド プロシージャに生成されます。

于 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 に答える