AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • Início
  • system&network
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • Início
  • system&network
    • Recentes
    • Highest score
    • tags
  • Ubuntu
    • Recentes
    • Highest score
    • tags
  • Unix
    • Recentes
    • tags
  • DBA
    • Recentes
    • tags
  • Computer
    • Recentes
    • tags
  • Coding
    • Recentes
    • tags
Início / dba / Perguntas / 282886
Accepted
Poseidon
Poseidon
Asked: 2021-01-10 01:30:40 +0800 CST2021-01-10 01:30:40 +0800 CST 2021-01-10 01:30:40 +0800 CST

Girando o conjunto de resultados de 4 por 44

  • 772

Eu tenho uma consulta SQL Server que obtém os dados de vendas para uma cadeia de mercado de 44 lojas, inclui 4 valores para cada loja (Vendas, CPV, GP e GP Margin) , declarei um CTE e entrei nele 44 vezes para cada loja obter os 4 valores como colunas para cada loja conforme mostrado abaixo:

Consulta: Consulta1

Resultado 1

mas quando tento usar PIVOTfunção com SQL dinâmico ele retorna vários nulos conforme mostrado abaixo:

Consulta: Consulta2 Resultado 2

table ##tbl1inclui o conjunto de dados que quero dinamizar: ##tbl1

Usei a seguinte consulta: (os três pontos representam o restante das colunas a serem dinamizadas)

select * from ##tbl1 

pivot (sum(total_sales)     for s in ([50001 Sales],[50002 Sales],...)) as pv_tb 
pivot (sum(Margin)          for m in ([50001 margin],[50002 margin],...)) as pv_tb1
pivot (sum(total_profit)    for p in ([50001 profit],[50002 profit],...)) as pv_tb2
pivot (sum(total_cost)      for c in ([50001 cost],[50002 cost],...)) as pv_tb3

Eu também uso SQL dinâmico para passar os nomes das colunas para dentro PIVOT()sem ter que escrever cada uma delas individualmente, mas por uma questão de solução de problemas eu apenas mostrei como a consulta ficaria.

Não consigo usar CROSS APPLYe UNPIVOTcom esse resultado de 132 colunas, é muito difícil manter.

Alguém pode me ajudar a encontrar uma maneira mais fácil do que isso?

sql-server sql-server-2012
  • 2 2 respostas
  • 57 Views

2 respostas

  • Voted
  1. nbk
    2021-01-10T06:13:05+08:002021-01-10T06:13:05+08:00
      WITH Sales_CTE ([50001 Sales], [50002 Sales], [50001 margin],[50002 margin],[50001 profit],[50002 profit],[50001 cost],[50002 cost])  
    AS  
    (    
    select * from 
        (
        select rtrim(cast(csstor as char))+' Sales' as s
                ,rtrim(cast(csstor as char))+' margin' as m
                ,rtrim(cast(csstor as char))+' profit' as p
                ,rtrim(cast(csstor as char))+' cost' as c
                ,[TOTAL_PRICE]
                ,[Margin] 
                ,[TOTAL_Profit]     
                ,[Total_COST]
        from sales
        
        ) as result_1
    pivot (sum(total_price)     for s in ([50001 Sales],[50002 Sales])) as pv_tb 
    pivot (sum(Margin)          for m in ([50001 margin],[50002 margin])) as pv_tb1
    pivot (sum(total_profit)    for p in ([50001 profit],[50002 profit])) as pv_tb2
    pivot (sum(total_cost)      for c in ([50001 cost],[50002 cost])) as pv_tb3)
    SELECT
    MAX([50001 Sales]) as [50001 Sales]
    , MAX([50002 Sales]) AS [50002 Sales]
    , MAX([50001 margin]) AS [50001 margin]
    ,MAX([50002 margin]) AS [50002 margin]
    ,MAX([50001 profit]) AS [50002 margin]
    ,MAX([50002 profit]) AS [50002 profit]
    ,MAX([50001 cost]) AS [50001 cost]
    ,MAX([50002 cost]) AS [50002 cost]
    FROM Sales_CTE
    

    O resultado é

    50001 Vendas 50002 Vendas 50001 margem 50002 margem 50002 margem 50002 lucro 50001 custo 50002 custo
    39498.300 31647.150 12.562.300 11.159300 4961.932 3531.624 34536,3675 28115,5264
    • 1
  2. Best Answer
    Jonathan Fite
    2022-06-25T05:09:22+08:002022-06-25T05:09:22+08:00

    Aqui está a minha solução dinâmica para esta questão. Acho que o problema que você está tendo com a instrução pivô é que todas as colunas estão presentes quando você gira, o que fornece esses NULLs errôneos. Mesmo que eles não estejam na lista de seleção, o operador de pivô agrupa por eles (veja como eu faço uma sub-seleção na seção de pivô).

    Quanto ao desempenho, sob o capô, a diferença de desempenho é insignificante, PIVOT e UNPIVOT são açúcar sintático sobre SQL aparentemente feio/complicado. Eu prefiro usá-los, pois acho que mantém o código mais fácil de ler com menos ruído.

    De qualquer forma, esta solução dá os resultados desejados e o faz dinamicamente. A única coisa que eu tive que mudar foram os tipos de dados. O operador UNPIVOT requer que todos sejam do mesmo tipo de dados, então eu apenas os forcei a serem NUMERIC(38,6), você pode ter que adicionar uma etapa de conversão/CTE à mistura.

    Uma coisa a notar é que ele não faz nenhuma filtragem da tabela de origem. Se você precisar adicionar filtragem, precisará atualizá-la em vários locais ou a consulta começará a fornecer esses resultados com os valores NULL. Melhor despejar os resultados desejados em uma tabela temporária nesse caso.

    Finalmente: Este é realmente um trabalho mais adequado para a camada de exibição. Não tenho certeza para onde os resultados estão indo, mas o SSRS e o Excel têm excelentes operações de pivô incorporadas e farão um trabalho melhor ao lidar com isso dinamicamente do que você pode fazer no SQL. Mesmo assim, minha solução:

    /** BUILD UP BASE TABLE
        -I made it a temp table, but you can do whatever.
        **/
    
    DROP TABLE IF EXISTS #Sales 
    
    CREATE TABLE #Sales (
        [CSSTOR] [numeric](5, 0) NOT NULL,
        [TOTAL_PRICE] [numeric](38, 6) NULL,
        [Margin] [numeric](38, 6) NULL,
        [TOTAL_Profit] [numeric](38, 6) NULL,
        [Total_COST] [numeric](38, 6) NULL
    ) 
    
    
    INSERT INTO #Sales VALUES(50036,    25131.250,  12.876400   ,3236.008   ,21895.2421)
    INSERT INTO #Sales VALUES(50026,    87732.470,  13.901600   ,12196.269  ,75536.2006)
    INSERT INTO #Sales VALUES(50006,    77530.620,  13.841900   ,10731.743  ,66798.8770)
    INSERT INTO #Sales VALUES(50017,    40097.190,  11.662900   ,4676.506   ,35420.6844)
    INSERT INTO #Sales VALUES(50047,    45848.710,  14.268400   ,6541.888   ,39306.8219)
    INSERT INTO #Sales VALUES(50037,    17029.810,  13.829900   ,2355.212   ,14674.5980)
    INSERT INTO #Sales VALUES(50027,    75493.600,  14.490900   ,10939.739  ,64553.8611)
    INSERT INTO #Sales VALUES(50007,    27153.030,  11.803900   ,3205.127   ,23947.9030)
    INSERT INTO #Sales VALUES(50048,    46728.060,  12.531400   ,5855.688   ,40872.3721)
    INSERT INTO #Sales VALUES(50028,    30872.940,  13.028200   ,4022.209   ,26850.7310)
    INSERT INTO #Sales VALUES(50008,    61974.180,  13.886100   ,8605.854   ,53368.3260)
    INSERT INTO #Sales VALUES(50038,    17190.610,  12.486200   ,2146.463   ,15044.1469)
    INSERT INTO #Sales VALUES(50039,    48018.250,  5.346900    ,2567.526   ,45450.7237)
    INSERT INTO #Sales VALUES(50049,    35941.500,  11.092800   ,3986.927   ,31954.5728)
    INSERT INTO #Sales VALUES(50029,    44106.940,  13.131800   ,5792.052   ,38314.8880)
    INSERT INTO #Sales VALUES(50019,    39339.940,  14.160600   ,5570.799   ,33769.1408)
    INSERT INTO #Sales VALUES(50040,    40693.330,  12.369900   ,5033.752   ,35659.5784)
    INSERT INTO #Sales VALUES(50030,    53085.970,  12.046700   ,6395.124   ,46690.8460)
    INSERT INTO #Sales VALUES(50020,    61528.750,  13.056700   ,8033.643   ,53495.1072)
    INSERT INTO #Sales VALUES(50010,    40194.330,  12.991800   ,5222.006   ,34972.3242)
    INSERT INTO #Sales VALUES(50041,    40097.640,  12.226900   ,4902.718   ,35194.9217)
    INSERT INTO #Sales VALUES(50031,    77884.530,  9.238900    ,7195.694   ,70688.8360)
    INSERT INTO #Sales VALUES(50021,    90946.760,  12.355500   ,11236.982  ,79709.7779)
    INSERT INTO #Sales VALUES(50001,    39498.300,  12.562300   ,4961.932   ,34536.3675)
    INSERT INTO #Sales VALUES(50042,    64223.200,  13.290500   ,8535.633   ,55687.5668)
    INSERT INTO #Sales VALUES(50032,    39490.130,  14.726000   ,5815.355   ,33674.7753)
    INSERT INTO #Sales VALUES(50022,    66602.810,  12.567800   ,8370.569   ,58232.2408)
    INSERT INTO #Sales VALUES(50012,    31329.510,  14.151300   ,4433.557   ,26895.9526)
    INSERT INTO #Sales VALUES(50002,    31647.150,  11.159300   ,3531.624   ,28115.5264)
    INSERT INTO #Sales VALUES(50043,    55461.870,  13.161100   ,7299.393   ,48162.4773)
    INSERT INTO #Sales VALUES(50033,    35486.720,  8.224400    ,2918.604   ,32568.1158)
    INSERT INTO #Sales VALUES(50023,    70549.250,  12.387900   ,8739.596   ,61809.6541)
    INSERT INTO #Sales VALUES(50013,    35080.480,  13.330000   ,4676.263   ,30404.2168)
    INSERT INTO #Sales VALUES(50003,    39691.180,  11.286800   ,4479.868   ,35211.3115)
    INSERT INTO #Sales VALUES(50044,    33908.310,  11.479500   ,3892.531   ,30015.7794)
    INSERT INTO #Sales VALUES(50034,    17208.780,  8.398300    ,1445.262   ,15763.5180)
    INSERT INTO #Sales VALUES(50024,    37125.040,  10.122100   ,3757.867   ,33367.1726)
    INSERT INTO #Sales VALUES(50014,    51475.620,  12.559800   ,6465.278   ,45010.3422)
    INSERT INTO #Sales VALUES(50004,    67062.810,  13.691500   ,9181.943   ,57880.8672)
    INSERT INTO #Sales VALUES(50045,    38778.360,  11.264400   ,4368.184   ,34410.1756)
    INSERT INTO #Sales VALUES(50035,    19578.640,  10.538400   ,2063.281   ,17515.3589)
    INSERT INTO #Sales VALUES(50025,    58158.960,  13.853000   ,8056.790   ,50102.1700)
    INSERT INTO #Sales VALUES(50015,    34844.170,  8.108700    ,2825.443   ,32018.7270)
    INSERT INTO #Sales VALUES(50005,    32089.110,  13.341200   ,4281.103   ,27808.0067)
    
    DECLARE @SQLCMD NVARCHAR(MAX)
    DECLARE @CHeaderList NVARCHAR(MAX)
    
    /** Build a list of headers in the correct order that we want (store order, then sales, margin, profit and cost).
        **/
    SET @CHeaderList = (SELECT STRING_AGG(QUOTENAME(CONVERT(NVARCHAR(MAX), CONCAT(CONVERT(NVARCHAR(20), CSSTOR), ' - ', [Type]))), ',') WITHIN GROUP (ORDER BY CONVERT(NVARCHAR(20), CSSTOR), [Sort])
                        FROM (  SELECT DISTINCT CSSTOR, [Type], [Sort]
                                FROM #Sales 
                                    CROSS APPLY (VALUES ('Sales', 1)
                                                        , ('Margin', 2)
                                                        , ('Profit', 3)
                                                        , ('Cost', 4)
                                                        ) AS P ([Type], [Sort])
                                                        ) AS D
                                                        )
    /** Build up our dynamic sql command.
        CTE_Unpvt - Unpivots the data to a store/type/value array. 
            It also builds up a "CHeader" column that matches the pattern in the CHeaderList.  This gives us '50047 - Cost'
            for example.
        
        CTE_PVT - Pivots the data back to what we want using the CHeaderList above.  
        
        **/
    
    SET @SQLCMD = N'
    ;WITH CTE_Unpvt AS
        (
        SELECT CSSTOR
            , [Type]
            , [Value]
            , CHeader = CONCAT(CONVERT(NVARCHAR(50), CSSTOR), '' - '', CASE WHEN [Type] = ''TOTAL_PRICE'' THEN ''Sales''
                                                                            WHEN [Type] = ''Margin'' THEN ''Margin''
                                                                            WHEN [Type] = ''TOTAL_Profit'' THEN ''Profit''
                                                                            WHEN [Type] = ''TOTAL_COST'' THEN ''Cost''
                                                                            ELSE '''' 
                                                                            END)
        FROM #Sales AS S
            UNPIVOT ([Value] FOR [Type] IN ([TOTAL_PRICE], [Margin], [TOTAL_Profit], [Total_COST])) AS Unpvt 
        )
    , CTE_PVT AS
        (
        SELECT ' + @CHeaderList + '
        FROM (SELECT [Value], CHeader FROM CTE_Unpvt AS p) AS up
            PIVOT (SUM([Value]) FOR CHeader IN (' + @CHeaderList + ')) AS pvt
        )
    SELECT * 
        , [TOTAL_SALES] = (SELECT SUM([TOTAL_PRICE]) FROM #Sales)
        , [TOTAL_MARGIN] = (SELECT SUM([Margin]) FROM #Sales)
        , [TOTAL_PROFIT] = (SELECT SUM([TOTAL_Profit]) FROM #Sales)
        , [TOTAL_COST] = (SELECT SUM([TOTAL_COST]) FROM #Sales)
    FROM CTE_PVT'
        
    EXEC (@SQLCMD)
    
    • 1

relate perguntas

  • SQL Server - Como as páginas de dados são armazenadas ao usar um índice clusterizado

  • Preciso de índices separados para cada tipo de consulta ou um índice de várias colunas funcionará?

  • Quando devo usar uma restrição exclusiva em vez de um índice exclusivo?

  • Quais são as principais causas de deadlocks e podem ser evitadas?

  • Como determinar se um Índice é necessário ou necessário

Sidebar

Stats

  • Perguntas 205573
  • respostas 270741
  • best respostas 135370
  • utilizador 68524
  • Highest score
  • respostas
  • Marko Smith

    conectar ao servidor PostgreSQL: FATAL: nenhuma entrada pg_hba.conf para o host

    • 12 respostas
  • Marko Smith

    Como fazer a saída do sqlplus aparecer em uma linha?

    • 3 respostas
  • Marko Smith

    Selecione qual tem data máxima ou data mais recente

    • 3 respostas
  • Marko Smith

    Como faço para listar todos os esquemas no PostgreSQL?

    • 4 respostas
  • Marko Smith

    Listar todas as colunas de uma tabela especificada

    • 5 respostas
  • Marko Smith

    Como usar o sqlplus para se conectar a um banco de dados Oracle localizado em outro host sem modificar meu próprio tnsnames.ora

    • 4 respostas
  • Marko Smith

    Como você mysqldump tabela (s) específica (s)?

    • 4 respostas
  • Marko Smith

    Listar os privilégios do banco de dados usando o psql

    • 10 respostas
  • Marko Smith

    Como inserir valores em uma tabela de uma consulta de seleção no PostgreSQL?

    • 4 respostas
  • Marko Smith

    Como faço para listar todos os bancos de dados e tabelas usando o psql?

    • 7 respostas
  • Martin Hope
    Jin conectar ao servidor PostgreSQL: FATAL: nenhuma entrada pg_hba.conf para o host 2014-12-02 02:54:58 +0800 CST
  • Martin Hope
    Stéphane Como faço para listar todos os esquemas no PostgreSQL? 2013-04-16 11:19:16 +0800 CST
  • Martin Hope
    Mike Walsh Por que o log de transações continua crescendo ou fica sem espaço? 2012-12-05 18:11:22 +0800 CST
  • Martin Hope
    Stephane Rolland Listar todas as colunas de uma tabela especificada 2012-08-14 04:44:44 +0800 CST
  • Martin Hope
    haxney O MySQL pode realizar consultas razoavelmente em bilhões de linhas? 2012-07-03 11:36:13 +0800 CST
  • Martin Hope
    qazwsx Como posso monitorar o andamento de uma importação de um arquivo .sql grande? 2012-05-03 08:54:41 +0800 CST
  • Martin Hope
    markdorison Como você mysqldump tabela (s) específica (s)? 2011-12-17 12:39:37 +0800 CST
  • Martin Hope
    Jonas Como posso cronometrar consultas SQL usando psql? 2011-06-04 02:22:54 +0800 CST
  • Martin Hope
    Jonas Como inserir valores em uma tabela de uma consulta de seleção no PostgreSQL? 2011-05-28 00:33:05 +0800 CST
  • Martin Hope
    Jonas Como faço para listar todos os bancos de dados e tabelas usando o psql? 2011-02-18 00:45:49 +0800 CST

Hot tag

sql-server mysql postgresql sql-server-2014 sql-server-2016 oracle sql-server-2008 database-design query-performance sql-server-2017

Explore

  • Início
  • Perguntas
    • Recentes
    • Highest score
  • tag
  • help

Footer

AskOverflow.Dev

About Us

  • About Us
  • Contact Us

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve