Rumah > pangkalan data > tutorial mysql > Bagaimana untuk Menjana Lajur secara Dinamik dalam SQL Menggunakan Fungsi PIVOT?

Bagaimana untuk Menjana Lajur secara Dinamik dalam SQL Menggunakan Fungsi PIVOT?

Susan Sarandon
Lepaskan: 2025-01-09 15:42:45
asal
279 orang telah melayarinya

How to Dynamically Generate Columns in SQL Using the PIVOT Function?

Menjana Lajur Secara Dinamik dalam SQL Menggunakan Fungsi PIVOT

Artikel ini menangani cabaran mencipta lajur dalam SQL secara dinamik untuk mewakili pelbagai jenis ganjaran dalam pangkalan data ganjaran pelanggan. Jenis ganjaran tidak tetap dan boleh berubah dari semasa ke semasa. Kami akan meneroka penyelesaian menggunakan fungsi SQL PIVOT, menunjukkan kedua-dua pendekatan statik dan dinamik, dan termasuk penambahan jumlah lajur untuk hasil yang komprehensif.

Masalahnya:

Kami mempunyai jadual pangkalan data untuk pelanggan, ganjaran mereka dan jenis ganjaran. Jenis ganjaran adalah dinamik; mereka boleh ditambah atau dialih keluar. Matlamatnya ialah untuk menjana set hasil dengan lajur untuk setiap jenis ganjaran, menunjukkan kiraan jenis ganjaran itu untuk setiap pelanggan.

Penyelesaian Statik (Bilangan Tetap Jenis Ganjaran):

Jika jenis ganjaran diketahui terlebih dahulu, pertanyaan PIVOT mudah sudah memadai:

<code class="language-sql">SELECT name, [Bronze], [Silver], [Gold], [Platinum], [AnotherOne]
FROM
(
  SELECT c.name, cr.description, r.typeid
  FROM customers c
  LEFT JOIN rewards r ON c.id = r.customerid
  LEFT JOIN customerrewards cr ON r.typeid = cr.typeid
) x
PIVOT
(
  COUNT(typeid)
  FOR description IN ([Bronze], [Silver], [Gold], [Platinum], [AnotherOne])
) p;</code>
Salin selepas log masuk

Pertanyaan ini berfungsi dengan baik, tetapi hanya jika jenis ganjaran ("Gangsa," "Perak," dll.) dipratakrifkan.

Penyelesaian Dinamik (Bilangan Jenis Ganjaran Tidak Diketahui):

Untuk bilangan jenis ganjaran yang dinamik, kami memerlukan SQL dinamik:

<code class="language-sql">DECLARE @cols AS NVARCHAR(MAX),
    @query  AS NVARCHAR(MAX);

SELECT @cols = STUFF((SELECT ',' + QUOTENAME(description)
                    FROM customerrewards
                    GROUP BY description, typeid
                    ORDER BY typeid
            FOR XML PATH(''), TYPE
            ).value('.', 'NVARCHAR(MAX)')
        ,1,1,'');

SET @query = 'SELECT name,' + @cols + ' FROM 
             (
                SELECT c.name, cr.description, r.typeid
                FROM customers c
                LEFT JOIN rewards r ON c.id = r.customerid
                LEFT JOIN customerrewards cr ON r.typeid = cr.typeid
            ) x
            PIVOT 
            (
                COUNT(typeid)
                FOR description IN (' + @cols + ')
            ) p;';

EXECUTE(@query);</code>
Salin selepas log masuk

Kod ini membina pertanyaan SQL secara dinamik, membina senarai lajur daripada jadual customerrewards. Ini membolehkan penyesuaian kepada sebarang jenis ganjaran.

Menambah Jumlah Lajur:

Untuk memasukkan jumlah kiraan ganjaran bagi setiap pelanggan, kami boleh menggunakan fungsi ROLLUP (kedua-dua versi statik dan dinamik ditunjukkan di bawah):

Versi Statik dengan Jumlah:

<code class="language-sql">SELECT name, SUM([Bronze]) Bronze, SUM([Silver]) Silver, 
  SUM([Gold]) Gold, SUM([Platinum]) Platinum, SUM([AnotherOne]) AnotherOne
FROM 
(
  SELECT name, [Bronze], [Silver], [Gold], [Platinum], [AnotherOne]
  FROM
  (
    SELECT c.name, cr.description, r.typeid
    FROM customers c
    LEFT JOIN rewards r ON c.id = r.customerid
    LEFT JOIN customerrewards cr ON r.typeid = cr.typeid
  ) x
  PIVOT
  (
    COUNT(typeid)
    FOR description IN ([Bronze], [Silver], [Gold], [Platinum], [AnotherOne])
  ) p
) x
GROUP BY name WITH ROLLUP;</code>
Salin selepas log masuk

Versi Dinamik dengan Jumlah:

<code class="language-sql">DECLARE @cols AS NVARCHAR(MAX),
    @colsRollup AS NVARCHAR(MAX),
    @query  AS NVARCHAR(MAX);

SELECT @cols = STUFF((SELECT ',' + QUOTENAME(description) 
                    FROM customerrewards
                    GROUP BY description, typeid
                    ORDER BY typeid
            FOR XML PATH(''), TYPE
            ).value('.', 'NVARCHAR(MAX)') 
        ,1,1,'');

SELECT @colsRollup 
      = STUFF((SELECT ', SUM(' + QUOTENAME(description) + ') AS ' + QUOTENAME(description)
                    FROM customerrewards
                    GROUP BY description, typeid
                    ORDER BY typeid
            FOR XML PATH(''), TYPE
            ).value('.', 'NVARCHAR(MAX)') 
        ,1,1,'');


SET @query 
          = 'SELECT name, ' + @colsRollup + '
             FROM
             (
                SELECT name,' + @cols + ' FROM 
                 (
                    SELECT c.name, cr.description, r.typeid
                    FROM customers c
                    LEFT JOIN rewards r ON c.id = r.customerid
                    LEFT JOIN customerrewards cr ON r.typeid = cr.typeid
                ) x
                PIVOT 
                (
                    COUNT(typeid)
                    FOR description IN (' + @cols + ')
                ) p 
              ) x1
              GROUP BY name WITH ROLLUP;';

EXECUTE(@query);</code>
Salin selepas log masuk

Contoh SQL dinamik ini menyediakan penyelesaian yang mantap dan fleksibel untuk menjana laporan dengan bilangan lajur jenis ganjaran yang berubah-ubah, termasuk jumlah lajur yang berguna. Ingat untuk menyesuaikan nama jadual dan lajur agar sepadan dengan skema pangkalan data khusus anda.

Atas ialah kandungan terperinci Bagaimana untuk Menjana Lajur secara Dinamik dalam SQL Menggunakan Fungsi PIVOT?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

sumber:php.cn
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Artikel terbaru oleh pengarang
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan