PostgreSQL
 sql >> Base de Dados >  >> RDS >> PostgreSQL

PostgreSQL converte colunas em linhas? Transpor?


Baseando minha resposta em uma tabela do formulário:
CREATE TABLE tbl (
   sl_no int
 , username text
 , designation text
 , salary int
);

Cada linha resulta em uma nova coluna para retornar. Com um tipo de retorno dinâmico como esse, dificilmente é possível tornar isso completamente dinâmico com uma única chamada ao banco de dados. Demonstrando soluções com duas etapas :
  1. Gerar consulta
  2. Executar consulta gerada

Geralmente, isso é limitado pelo número máximo de colunas que uma tabela pode conter. Portanto, não é uma opção para tabelas com mais de 1600 linhas (ou menos). Detalhes:
  • Qual ​​é o número máximo de colunas em uma consulta de seleção do PostgreSQL

Postgres 9.3 ou anterior

Solução dinâmica com crosstab()

  • Completamente dinâmico, funciona para qualquer mesa. Forneça o nome da tabela em dois lugares:
SELECT 'SELECT *
FROM   crosstab(
       ''SELECT unnest(''' || quote_literal(array_agg(attname))
                           || '''::text[]) AS col
             , row_number() OVER ()
             , unnest(ARRAY[' || string_agg(quote_ident(attname)
                              || '::text', ',') || ']) AS val
        FROM   ' || attrelid::regclass || '
        ORDER  BY generate_series(1,' || count(*) || '), 2''
   ) t (col text, '
     || (SELECT string_agg('r'|| rn ||' text', ',')
         FROM (SELECT row_number() OVER () AS rn FROM tbl) t)
     || ')' AS sql
FROM   pg_attribute
WHERE  attrelid = 'tbl'::regclass
AND    attnum > 0
AND    NOT attisdropped
GROUP  BY attrelid;

Poderia ser encapsulado em uma função com um único parâmetro ...
Gera uma consulta da forma:
SELECT *
FROM   crosstab(
       'SELECT unnest(''{sl_no,username,designation,salary}''::text[]) AS col
             , row_number() OVER ()
             , unnest(ARRAY[sl_no::text,username::text,designation::text,salary::text]) AS val
        FROM   tbl
        ORDER  BY generate_series(1,4), 2'
   ) t (col text, r1 text,r2 text,r3 text,r4 text)

Produz o resultado desejado:
col         r1    r2      r3     r4
-----------------------------------
sl_no       1      2      3      4
username    A      B      C      D
designation XYZ    RTS    QWE    HGD
salary      10000  50000  20000  34343

Solução simples com unnest()

SELECT 'SELECT unnest(''{sl_no, username, designation, salary}''::text[] AS col)
     , ' || string_agg('unnest('
                    || quote_literal(ARRAY[sl_no::text, username::text, designation::text, salary::text])
                    || '::text[]) AS row' || sl_no, E'\n     , ') AS sql
FROM   tbl;
  • Lento para tabelas com mais de duas colunas.

Gera uma consulta do formulário:
SELECT unnest('{sl_no, username, designation, salary}'::text[]) AS col
     , unnest('{10,Joe,Music,1234}'::text[]) AS row1
     , unnest('{11,Bob,Movie,2345}'::text[]) AS row2
     , unnest('{12,Dave,Theatre,2356}'::text[]) AS row3
     , unnest('{4,D,HGD,34343}'::text[]) AS row4

Mesmo resultado.

Postgres 9.4+

Solução dinâmica com crosstab()


Use isso se puder. Bate o resto.
SELECT 'SELECT *
FROM   crosstab(
       $ct$SELECT u.attnum, t.rn, u.val
        FROM  (SELECT row_number() OVER () AS rn, * FROM '
                              || attrelid::regclass || ') t
             , unnest(ARRAY[' || string_agg(quote_ident(attname)
                              || '::text', ',') || '])
                 WITH ORDINALITY u(val, attnum)
        ORDER  BY 1, 2$ct$
   ) t (attnum bigint, '
     || (SELECT string_agg('r'|| rn ||' text', ', ')
         FROM  (SELECT row_number() OVER () AS rn FROM tbl) t)
     || ')' AS sql
FROM   pg_attribute
WHERE  attrelid = 'tbl'::regclass
AND    attnum > 0
AND    NOT attisdropped
GROUP  BY attrelid;

Operando com attnum em vez de nomes de colunas reais. Mais simples e rápido. Junte o resultado a pg_attribute mais uma vez ou integre os nomes das colunas como no exemplo da página 9.3.
Gera uma consulta do formato:
SELECT *
FROM   crosstab(
       $ct$SELECT u.attnum, t.rn, u.val
        FROM  (SELECT row_number() OVER () AS rn, * FROM tbl) t
             , unnest(ARRAY[sl_no::text,username::text,designation::text,salary::text])
                WITH ORDINALITY u(val, attnum)
        ORDER  BY 1, 2$ct$
   ) t (attnum bigint, r1 text, r2 text, r3 text, r4 text);

Isso usa toda uma gama de recursos avançados. Simplesmente demais para explicar.

Solução simples com unnest()


Um unnest() agora pode levar vários arrays para desaninhar em paralelo.
SELECT 'SELECT * FROM unnest(
  ''{sl_no, username, designation, salary}''::text[]
, ' || string_agg(quote_literal(ARRAY[sl_no::text, username::text, designation::text, salary::text])
              || '::text[]', E'\n, ')
    || E') \n AS t(col,' || string_agg('row' || sl_no, ',') || ')' AS sql
FROM   tbl;

Resultado:
SELECT * FROM unnest(
 '{sl_no, username, designation, salary}'::text[]
,'{10,Joe,Music,1234}'::text[]
,'{11,Bob,Movie,2345}'::text[]
,'{12,Dave,Theatre,2356}'::text[])
 AS t(col,row1,row2,row3,row4)

Fiddle SQL rodando na página 9.3.