我在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知识非常平庸,我根本不知道如何在一个选择中将每列作为一行获取。
有人能帮忙吗?提前谢谢。
答案 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
替换存储过程。