问题描述
我正在尝试在 postgresql 中编写函数,该函数创建包含 table_name 文本、table_rec jsonb 列的 temp_table,并使用包含 json 中表和记录名称的表中的表名填充 for 循环。我在字符串中有 for 循环,我想执行它。 但它不起作用。 我有变量 rec 记录、sql_query 文本和 tab_name 文本,我想这样做:
CREATE OR REPLACE FUNCTION public.test51(
)
RETURNS TABLE(tabel_name text,record_json jsonb)
LANGUAGE 'plpgsql'
COST 100
VOLATILE
ROWS 1000
AS $BODY$
declare
rec record;
tabel_name text;
tabel_names text[];
counter integer := 1;
sql_query text;
limit_for_sending integer;
rec_count integer;
begin
select into tabel_names array(select "TABLE_NAME" from public."TABLES");
create temp table temp_tab(tab_nam text,recik jsonb);
while array_length(tabel_names,1) >= counter loop
tabel_name := '"' || tabel_names[counter] || '"';
select into limit_for_sending "TABLE_LIMIT_FOR_SENDING_DATA" from public."TABLES" where "TABLE_NAME" = tabel_name;
sql_query := 'select count(*) from public.' || tabel_name;
execute sql_query into rec_count;
if (rec_count >= limit_for_sending and limit_for_sending is not null) then
sql_query := 'for rec in select * from public.' || tabel_name || '
loop
insert into temp_tab
select ' || tabel_name || ',to_jsonb(rec);
end loop';
execute sql_query;
end if;
counter := counter + 1;
end loop;
return query
select * from temp_tabik;
drop table temp_tabik;
end;
$BODY$;
感谢您的回复。
解决方法
您似乎有一些表,其中包含您希望将所有行作为 JSONB 返回的表的信息。并且该元表还包含一列,该列设置了不应返回行的阈值。
您不需要临时表或数组来存储表名。您可以遍历 TABLES
表上的查询并直接在该循环中运行动态 SQL。
return query
不终止函数,它只是将查询的结果附加到函数的结果中。
动态 SQL 最好使用 format()
函数创建,因为它更易于阅读并且使用 %I
占位符将正确处理带引号的标识符(这非常重要,因为您正在使用那些可怕的大写表名)
据我所知,您的功能可以简化为:
CREATE OR REPLACE FUNCTION public.test51()
RETURNS TABLE(tabel_name text,record_json jsonb)
LANGUAGE plpgsql
AS
$BODY$
declare
rec record;
sql_query text;
rec_count bigint;
begin
for rec in
select "TABLE_NAME" as table_name,"TABLE_LIMIT_FOR_SENDING_DATA" as rec_limit
from public."TABLES"
loop
if rec.rec_limit is not null then
execute format('select count(*) from %I',rec.table_name)
into rec_count;
end if;
if (rec.rec_limit is not null and rec_count >= rec.rec_limit) then
sql_query := format('select %L,to_jsonb(t) from %I as t',rec.table_name,rec.table_name);
return query execute sql_query;
end if;
end loop;
end;
$BODY$;
一些注意事项
- 语言名称是一个标识符,不应用单引号括起来。此语法已弃用,可能会在未来版本中删除,因此请不要习惯它。
- 你真的应该避免那些可怕的带引号的标识符。他们带来的麻烦远远超过他们的价值。有关详细信息,请参阅 the Postgres wiki。