sql >> Base de Datos >  >> RDS >> Oracle

Oracle:nombre de columna dinámica

Dado que está utilizando Oracle10g, no hay PIVOT por lo que tendrá que realizar este tipo de transformación usando una función agregada con un CASE declaración.

Si los valores se conocen con anticipación, puede codificarlos en una versión estática:

select s.ts_location,
  sum(case when p.tp_name = 'apple' then s.ts_sales else 0 end) Apple,
  sum(case when p.tp_name = 'mango' then s.ts_sales else 0 end) Mango,
  sum(case when p.tp_name = 'pineapple' then s.ts_sales else 0 end) Pineapple
from tbl_sales s
inner join tbl_products p
  on s.ts_tp_id = p.tp_id
group by s.ts_location

Consulte SQL Fiddle con demostración

Pero si sus valores no se conocen con anticipación, entonces debe implementar sql dinámico y en Oracle querrá usar un procedimiento para esto:

CREATE OR REPLACE procedure dynamic_pivot(p_cursor in out sys_refcursor)
as
    sql_query varchar2(1000) := 'select s.ts_location ';

    begin
        for x in (select distinct tp_name from tbl_products order by 1)
        loop
            sql_query := sql_query ||
                ' , sum(case when p.tp_name = '''||x.tp_name||''' then s.ts_sales end) as '||x.tp_name;

                dbms_output.put_line(sql_query);
        end loop;

        sql_query := sql_query || ' from tbl_sales s 
                                                inner join tbl_products p
                                                  on s.ts_tp_id = p.tp_id
                                                group by s.ts_location';
        dbms_output.put_line(sql_query);

        open p_cursor for sql_query;
    end;
/

Luego, para devolver los resultados, puede usar (nota: así es como lo hago en Toad):

variable x refcursor
exec dynamic_pivot(:x)
print x

Ambos devolverán el resultado:

| TS_LOCATION | APPLE | MANGO | PINEAPPLE |
-------------------------------------------
|          LN |     0 |    10 |        35 |
|          QL |    25 |     0 |        20 |
|          NY |   100 |     5 |        50 |