在Firebird 2.5中转置列和行

时间:2018-06-06 15:15:17

标签: firebird transpose unions firebird2.5

我在Firebird(方言3)中编写了一个程序,它返回了类似这样的内容:

 column1  |  column2    |  column3  |   column4  |  ...
----------|-------------|-----------|------------|--------
 1        |  55         |  2.5      |    100€    |  ...

具体的列名称并不重要。我像这样访问它

SELECT * FROM MY_PROCEDURE(:START_DATE, :END_DATE);

它只返回一行,所以我想我也可以用EXECUTE_PROCEDURE访问它。 现在我想要的是在返回

中转置列和行
 row      |  result    
----------|---------
 column1  |  1    
 column2  |  55    
 column3  |  2.0   
 column4  |  100€    
 ...      |  ...  

我最初做的是这样的想法:

select 'column1' AS row, column1 AS result
FROM MY_PROCEDURE(:START_DATE, :END_DATE)
union all
select 'column2' AS row, column2 AS result
FROM MY_PROCEDURE(:START_DATE, :END_DATE)
union all
...

基本上每列都有一个查询。有效。但是,最终我遇到了这个问题:

Dynamic SQL Error
Too many Contexts of Relation/Procedure/Views. Maxium allowed is 255.

所以我需要重构我的脚本。正如您所看到的,我的SQL知识非常平庸,我根本不知道如何在一个选择中将每列作为一行获取。

有人能帮忙吗?提前谢谢。

1 个答案:

答案 0 :(得分:1)

Firebird本身没有unpivot或其他内置支持转置列。

“最佳”解决方案,可能性能最佳的解决方案是重写MY_PROCEDURE(或编写替代版本)以输出转置的行。

例如,假设您的存储过程执行以下操作:

set term #;
create procedure test_1 
    returns (id integer, column1 double precision, column2 double precision, column3 double precision)
as
begin 
    for 
        select id, column1, column2, column3 
        from sometable 
        into :id, :column1, :column2, :column3 do
    begin
        suspend;
    end
end#
set term ;#

然后,您可以通过手动将值转换为单独的挂起来重写此内容:

set term #;
create procedure test_2 
    returns (id integer, columnname varchar(100), columnvalue double precision)
as
declare column1 double precision;
declare column2 double precision;
declare column3 double precision;
begin 
    for 
        select id, column1, column2, column3 
        from sometable 
        into :id, :column1, :column2, :column3 do
    begin
        columnname = 'column1';
        columnvalue = column1;
        suspend;
        columnname = 'column2';
        columnvalue = column2;
        suspend;
        columnname = 'column3';
        columnvalue = column3;
        suspend;
    end
end#
set term ;#

这将输出类似

的内容
id  columnname columnvalue
1   column1    1.0
1   column2    1.5
1   column3    5.0
2   ...etc

此解决方案确实要求所有输出(columnvalue)具有相同的类型。否则,您将需要转换为通用数据类型。

或者,您可以使用for select * from test_1 into ...将第一个过程链接到第二个过程。这可能或多或少有效,具体取决于存储过程的内部结构:

set term #;
create procedure test_3 
        returns (id integer, columnname varchar(100), columnvalue double precision)
as
declare column1 double precision;
declare column2 double precision;
declare column3 double precision;
begin 
    for 
        select id, column1, column2, column3 from test_1
        into :id, :column1, :column2, :column3 do
    begin
        columnname = 'column1';
        columnvalue = column1;
        suspend;
        columnname = 'column2';
        columnvalue = column2;
        suspend;
        columnname = 'column3';
        columnvalue = column3;
        suspend;
    end
end#
set term ;#

如果您需要输出的两种变体,那么最后一个选项可能是最好的,因为这意味着您只能为该存储过程的逻辑提供单一位置。

对于临时查询,您还可以使用具有相同代码的execute block替换存储过程。