postgresql for 文本形式的循环脚本无法执行

问题描述

我正在尝试在 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。

PL/pgSQL 中的

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

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...