SQL Serverの動的PIVOTクエリ?


202

私は次のデータを翻訳する手段を考え出す必要があります。

date        category        amount
1/1/2012    ABC             1000.00
2/1/2012    DEF             500.00
2/1/2012    GHI             800.00
2/10/2012   DEF             700.00
3/1/2012    ABC             1100.00

以下に:

date        ABC             DEF             GHI
1/1/2012    1000.00
2/1/2012                    500.00
2/1/2012                                    800.00
2/10/2012                   700.00
3/1/2012    1100.00

空白のスポットはNULLまたは空白のどちらでもかまいませんが、カテゴリは動的である必要があります。これに対するもう1つの可能性のある警告は、制限された容量でクエリを実行することです。つまり、一時テーブルが停止します。私は研究を試みて上陸しましたPIVOTが、それを理解するための最善の努力にもかかわらず、私は本当にそれを理解する前にそれを使用したことがないので、それを理解していません。誰かが私を正しい方向に向けることができますか?


3
SQL Serverのバージョンを教えてください。
アーロンベルトラン

回答:


250

動的SQL PIVOT:

create table temp
(
    date datetime,
    category varchar(3),
    amount money
)

insert into temp values ('1/1/2012', 'ABC', 1000.00)
insert into temp values ('2/1/2012', 'DEF', 500.00)
insert into temp values ('2/1/2012', 'GHI', 800.00)
insert into temp values ('2/10/2012', 'DEF', 700.00)
insert into temp values ('3/1/2012', 'ABC', 1100.00)


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

SET @cols = STUFF((SELECT distinct ',' + QUOTENAME(c.category) 
            FROM temp c
            FOR XML PATH(''), TYPE
            ).value('.', 'NVARCHAR(MAX)') 
        ,1,1,'')

set @query = 'SELECT date, ' + @cols + ' from 
            (
                select date
                    , amount
                    , category
                from temp
           ) x
            pivot 
            (
                 max(amount)
                for category in (' + @cols + ')
            ) p '


execute(@query)

drop table temp

結果:

Date                        ABC         DEF    GHI
2012-01-01 00:00:00.000     1000.00     NULL    NULL
2012-02-01 00:00:00.000     NULL        500.00  800.00
2012-02-10 00:00:00.000     NULL        700.00  NULL
2012-03-01 00:00:00.000     1100.00     NULL    NULL

したがって、\ @ colsは文字列連結である必要がありますよね?そこに\ @colsを挿入するためにsp_executesqlとパラメーターバインディングを使用することはできませんか?\ @colsを自分で作成したとしても、悪意のあるSQLが含まれているとしたらどうでしょう。それを連結して実行する前に実行できる追加の緩和策はありますか?
Red Pea

これで行と列をどのように並べ替えますか?
Patrick Schomburg、2016年

@PatrickSchomburgさまざまな方法があります。並べ替えを行う@cols場合は、を削除しDISTINCTて使用GROUP BYORDER BY、のリストを取得することができます@cols
タリン

やってみます。行はどうですか?日付も使っていますが、順番が出ません。
Patrick Schomburg、2016年

1
私が間違った場所に注文したのは気にしないでください。
Patrick Schomburg、2016年

27

動的SQL PIVOT

列文字列を作成するための異なるアプローチ

create table #temp
(
    date datetime,
    category varchar(3),
    amount money
)

insert into #temp values ('1/1/2012', 'ABC', 1000.00)
insert into #temp values ('2/1/2012', 'DEF', 500.00)
insert into #temp values ('2/1/2012', 'GHI', 800.00)
insert into #temp values ('2/10/2012', 'DEF', 700.00)
insert into #temp values ('3/1/2012', 'ABC', 1100.00)

DECLARE @cols  AS NVARCHAR(MAX)='';
DECLARE @query AS NVARCHAR(MAX)='';

SELECT @cols = @cols + QUOTENAME(category) + ',' FROM (select distinct category from #temp ) as tmp
select @cols = substring(@cols, 0, len(@cols)) --trim "," at end

set @query = 
'SELECT * from 
(
    select date, amount, category from #temp
) src
pivot 
(
    max(amount) for category in (' + @cols + ')
) piv'

execute(@query)
drop table #temp

結果

date                    ABC     DEF     GHI
2012-01-01 00:00:00.000 1000.00 NULL    NULL
2012-02-01 00:00:00.000 NULL    500.00  800.00
2012-02-10 00:00:00.000 NULL    700.00  NULL
2012-03-01 00:00:00.000 1100.00 NULL    NULL

13

私はこの質問が古いことを知っていますが、私は答えを探していて、問題の「動的」な部分を拡張して、誰かを助けることができるかもしれないと思いました。

何よりもまず、私がこのソリューションを構築して、数人の同僚が、すばやくピボットする必要のある一貫性のない大きなデータセットで抱えていた問題を解決しました。

このソリューションでは、ストアドプロシージャを作成する必要があるため、それがニーズに合わない場合は、今すぐ読み進めないでください。

この手順では、ピボットステートメントの主要な変数を取り込んで、さまざまなテーブル、列名、集計のピボットステートメントを動的に作成します。静的列は、ピボットのgroup by / identity列として使用されます(これは、必要でない場合はコードから取り除くことができますが、ピボットステートメントではかなり一般的であり、元の問題を解決するために必要でした)。ピボット列は、最終的な結果の列名が生成され、値列は集計が適用されるものです。Tableパラメータは、スキーマを含むテーブルの名前(schema.tablename)です。コードのこの部分は、希望するほどクリーンではないため、多少の愛情を抱く可能性があります。私の使用法は公にされておらず、SQLインジェクションは問題ではなかったので、私にとってはうまくいきました。

ストアドプロシージャを作成するコードから始めましょう。このコードはSSMS 2005以降のすべてのバージョンで機能するはずですが、2005または2016ではテストしていませんが、なぜ機能しないのかわかりません。

create PROCEDURE [dbo].[USP_DYNAMIC_PIVOT]
    (
        @STATIC_COLUMN VARCHAR(255),
        @PIVOT_COLUMN VARCHAR(255),
        @VALUE_COLUMN VARCHAR(255),
        @TABLE VARCHAR(255),
        @AGGREGATE VARCHAR(20) = null
    )

AS


BEGIN

SET NOCOUNT ON;
declare @AVAIABLE_TO_PIVOT NVARCHAR(MAX),
        @SQLSTRING NVARCHAR(MAX),
        @PIVOT_SQL_STRING NVARCHAR(MAX),
        @TEMPVARCOLUMNS NVARCHAR(MAX),
        @TABLESQL NVARCHAR(MAX)

if isnull(@AGGREGATE,'') = '' 
    begin
        SET @AGGREGATE = 'MAX'
    end


 SET @PIVOT_SQL_STRING =    'SELECT top 1 STUFF((SELECT distinct '', '' + CAST(''[''+CONVERT(VARCHAR,'+ @PIVOT_COLUMN+')+'']''  AS VARCHAR(50)) [text()]
                            FROM '+@TABLE+'
                            WHERE ISNULL('+@PIVOT_COLUMN+','''') <> ''''
                            FOR XML PATH(''''), TYPE)
                            .value(''.'',''NVARCHAR(MAX)''),1,2,'' '') as PIVOT_VALUES
                            from '+@TABLE+' ma
                            ORDER BY ' + @PIVOT_COLUMN + ''

declare @TAB AS TABLE(COL NVARCHAR(MAX) )

INSERT INTO @TAB EXEC SP_EXECUTESQL  @PIVOT_SQL_STRING, @AVAIABLE_TO_PIVOT 

SET @AVAIABLE_TO_PIVOT = (SELECT * FROM @TAB)


SET @TEMPVARCOLUMNS = (SELECT replace(@AVAIABLE_TO_PIVOT,',',' nvarchar(255) null,') + ' nvarchar(255) null')


SET @SQLSTRING = 'DECLARE @RETURN_TABLE TABLE ('+@STATIC_COLUMN+' NVARCHAR(255) NULL,'+@TEMPVARCOLUMNS+')  
                    INSERT INTO @RETURN_TABLE('+@STATIC_COLUMN+','+@AVAIABLE_TO_PIVOT+')

                    select * from (
                    SELECT ' + @STATIC_COLUMN + ' , ' + @PIVOT_COLUMN + ', ' + @VALUE_COLUMN + ' FROM '+@TABLE+' ) a

                    PIVOT
                    (
                    '+@AGGREGATE+'('+@VALUE_COLUMN+')
                    FOR '+@PIVOT_COLUMN+' IN ('+@AVAIABLE_TO_PIVOT+')
                    ) piv

                    SELECT * FROM @RETURN_TABLE'



EXEC SP_EXECUTESQL @SQLSTRING

END

次に、例のためにデータを準備します。受け入れられた回答からデータの例を取り、この概念実証で使用する2つのデータ要素を追加して、集計変更のさまざまな出力を示しています。

create table temp
(
    date datetime,
    category varchar(3),
    amount money
)

insert into temp values ('1/1/2012', 'ABC', 1000.00)
insert into temp values ('1/1/2012', 'ABC', 2000.00) -- added
insert into temp values ('2/1/2012', 'DEF', 500.00)
insert into temp values ('2/1/2012', 'DEF', 1500.00) -- added
insert into temp values ('2/1/2012', 'GHI', 800.00)
insert into temp values ('2/10/2012', 'DEF', 700.00)
insert into temp values ('2/10/2012', 'DEF', 800.00) -- addded
insert into temp values ('3/1/2012', 'ABC', 1100.00)

次の例は、簡単な例としてさまざまな集計を示すさまざまな実行ステートメントを示しています。例を簡単にするために、静的列、ピボット列、値列を変更することはしませんでした。コードをコピーして貼り付けるだけで、自分でいじり始めることができるはずです

exec [dbo].[USP_DYNAMIC_PIVOT] 'date','category','amount','dbo.temp','sum'
exec [dbo].[USP_DYNAMIC_PIVOT] 'date','category','amount','dbo.temp','max'
exec [dbo].[USP_DYNAMIC_PIVOT] 'date','category','amount','dbo.temp','avg'
exec [dbo].[USP_DYNAMIC_PIVOT] 'date','category','amount','dbo.temp','min'

この実行により、それぞれ次のデータセットが返されます。

ここに画像の説明を入力してください


よくやった!ストアドプロシージャの代わりにTVFのオプションを作成してください。そんなTVFから選ぶと便利でしょう。
Przemyslaw Remin 2017

3
残念ながら、私の知る限りではありません。TVFの動的な構造を持つことができないためです。TVFには静的な列のセットが必要です。
SFrejofsky 2017

8

STRING_AGG関数を使用してピボット列リストを作成するSQL Server 2017の更新バージョン:

create table temp
(
    date datetime,
    category varchar(3),
    amount money
);

insert into temp values ('20120101', 'ABC', 1000.00);
insert into temp values ('20120201', 'DEF', 500.00);
insert into temp values ('20120201', 'GHI', 800.00);
insert into temp values ('20120210', 'DEF', 700.00);
insert into temp values ('20120301', 'ABC', 1100.00);


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

SET @cols = (SELECT STRING_AGG(category,',') FROM (SELECT DISTINCT category FROM temp WHERE category IS NOT NULL)t);

set @query = 'SELECT date, ' + @cols + ' from 
            (
                select date
                    , amount
                    , category
                from temp
           ) x
            pivot 
            (
                 max(amount)
                for category in (' + @cols + ')
            ) p ';

execute(@query);

drop table temp;

6

これは、動的TSQLを使用して実現できます(SQLインジェクション攻撃を回避するには、QUOTENAMEを使用することを忘れないでください)。

SQL Server 2005の動的列を使用したピボット

SQL Server-動的PIVOTテーブル-SQLインジェクション

動的SQLの呪いと祝福義務的な言及


11
FWIW QUOTENAMEは、ユーザーからのパラメーターとして@tableNameを受け入れ、それをのようなクエリに追加する場合にのみ、SQLインジェクション攻撃に役立ちますSET @sql = 'SELECT * FROM ' + @tableName;。脆弱な動的SQL文字列をたっぷり構築できますQUOTENAMEが、なかなか役に立たないでしょう。
アーロンバートランド

2
@davids このメタディスカッションを参照しください。ハイパーリンクを削除すると、回答が不完全になります。
カーミット2014

@Kermit、私はコードを表示することがより役立つことに同意しますが、それが答えになるために必要であると言っていますか?リンクがなければ、私の回答は「動的TSQLを使用してこれを実現できます」です。選択された回答は同じルートを提案しますが、その方法も示す場合は利点が追加されるため、回答として選択されました。
davids 14

2
私は選択された回答(選択される前に)に賛成票を投じました。これは例があり、新しい人をよりよく助けるためです。ただし、新しい人も私が提供したリンクを読む必要があると思うので、削除しませんでした。
davids 14

3

不要なnull値をクリーンアップする私の解決策があります

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

select @cols = STUFF((SELECT ',' + QUOTENAME(CodigoFormaPago) 
                from PO_FormasPago
                order by CodigoFormaPago
        FOR XML PATH(''), TYPE
        ).value('.', 'NVARCHAR(MAX)') 
    ,1,1,'')

select @maxcols = STUFF((SELECT ',MAX(' + QUOTENAME(CodigoFormaPago) + ') as ' + QUOTENAME(CodigoFormaPago)
                from PO_FormasPago
                order by CodigoFormaPago
        FOR XML PATH(''), TYPE
        ).value('.', 'NVARCHAR(MAX)')
    ,1,1,'')

set @query = 'SELECT CodigoProducto, DenominacionProducto, ' + @maxcols + '
            FROM
            (
                SELECT 
                CodigoProducto, DenominacionProducto,
                ' + @cols + ' from 
                 (
                    SELECT 
                        p.CodigoProducto as CodigoProducto,
                        p.DenominacionProducto as DenominacionProducto,
                        fpp.CantidadCuotas as CantidadCuotas,
                        fpp.IdFormaPago as IdFormaPago,
                        fp.CodigoFormaPago as CodigoFormaPago
                    FROM
                        PR_Producto p
                        LEFT JOIN PR_FormasPagoProducto fpp
                            ON fpp.IdProducto = p.IdProducto
                        LEFT JOIN PO_FormasPago fp
                            ON fpp.IdFormaPago = fp.IdFormaPago
                ) xp
                pivot 
                (
                    MAX(CantidadCuotas)
                    for CodigoFormaPago in (' + @cols + ')
                ) p 
            )  xx 
            GROUP BY CodigoProducto, DenominacionProducto'

t @query;

execute(@query);

2

以下のコードは、NULLゼロに置き換える結果を提供します、出力でにします。

テーブルの作成とデータの挿入:

create table test_table
 (
 date nvarchar(10),
 category char(3),
 amount money
 )

 insert into test_table values ('1/1/2012','ABC',1000.00)
 insert into test_table values ('2/1/2012','DEF',500.00)
 insert into test_table values ('2/1/2012','GHI',800.00)
 insert into test_table values ('2/10/2012','DEF',700.00)
 insert into test_table values ('3/1/2012','ABC',1100.00)

NULLをゼロに置き換える正確な結果を生成するクエリ:

DECLARE @DynamicPivotQuery AS NVARCHAR(MAX),
@PivotColumnNames AS NVARCHAR(MAX),
@PivotSelectColumnNames AS NVARCHAR(MAX)

--Get distinct values of the PIVOT Column
SELECT @PivotColumnNames= ISNULL(@PivotColumnNames + ',','')
+ QUOTENAME(category)
FROM (SELECT DISTINCT category FROM test_table) AS cat

--Get distinct values of the PIVOT Column with isnull
SELECT @PivotSelectColumnNames 
= ISNULL(@PivotSelectColumnNames + ',','')
+ 'ISNULL(' + QUOTENAME(category) + ', 0) AS '
+ QUOTENAME(category)
FROM (SELECT DISTINCT category FROM test_table) AS cat

--Prepare the PIVOT query using the dynamic 
SET @DynamicPivotQuery = 
N'SELECT date, ' + @PivotSelectColumnNames + '
FROM test_table
pivot(sum(amount) for category in (' + @PivotColumnNames + ')) as pvt';

--Execute the Dynamic Pivot Query
EXEC sp_executesql @DynamicPivotQuery

出力:

ここに画像の説明を入力してください

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.