如何遍历 Oracle PLSQL 中的分隔列表

Posted

技术标签:

【中文标题】如何遍历 Oracle PLSQL 中的分隔列表【英文标题】:How to loop through a delimited list in Oracle PLSQL 【发布时间】:2013-10-03 20:38:13 【问题描述】:

我正在开发一个在其中调用另一个过程的 Oracle 过程。我的参数之一(parm1) 可以在逗号分隔列表中包含一个或多个值。如何循环遍历这些值以将它们一次传递给另一个过程?

这是我想要它做的一个例子:

When Parm1 = 123,312

callProcedure2(123)
callProcedure2(321)

-或-

When Parm1 123

callProcedure2(123)

我认为这可以使用循环来完成,但我不知道如何让它在循环中使用每个值作为单独的调用。

任何帮助将不胜感激!

谢谢!

【问题讨论】:

尝试寻找[Oracle] and split 这个question 看起来很有帮助 【参考方案1】:
CURSOR V_CUR IS
select regexp_substr(Parm1 ,'[^,]+', 1, level) As str from dual
connect by regexp_substr(Parm1, '[^,]+', 1, level) is not null;

这个游标会给你这样的结果

123
321

现在迭代游标并循环调用过程。

For i IN V_CUR
LOOP
    callProdcedure2(i.str);
END LOOP;

【讨论】:

谢谢!!这是一个很好的解决方案 这正是我想要的。【参考方案2】:

只需遍历子字符串:

declare 
  parm1 varchar2(1000) := '123,234,345,456,567,789,890';

  vStartIdx binary_integer;
  vEndIdx   binary_integer;
  vCurValue varchar2(1000);
begin

  vStartIdx := 0;
  vEndIdx   := instr(parm1, ','); 

  while(vEndIdx > 0) loop
    vCurValue := substr(parm1, vStartIdx+1, vEndIdx - vStartIdx - 1);

    -- call proc here
    dbms_output.put_line('->'||vCurValue||'<-');

    vStartIdx := vEndIdx;
    vEndIdx := instr(parm1, ',', vStartIdx + 1);  
  end loop;

  -- Call proc here for last part (or in case of single element)
  vCurValue := substr(parm1, vStartIdx+1);
  dbms_output.put_line('->'||vCurValue||'<-');

end;

【讨论】:

【参考方案3】:

有一个实用程序COMMA_TO_TABLE 和数组类型DBMS_UTILITY.UNCL_ARRAY 专门用于此任务。从 Oracle 10g 开始。

这是很好的文档here。

这是一个示例解决方案:

SET SERVEROUTPUT ON
DECLARE
    csvListElm VARCHAR2(4000) := 'elm1, elm2,elm3 ,elm4 , elm5';
    csvListTable DBMS_UTILITY.UNCL_ARRAY;
    csvListLen BINARY_INTEGER;
    currTableName VARCHAR2(222);
BEGIN
    DBMS_UTILITY.COMMA_TO_TABLE(csvListElm, csvListLen, csvListTable);
    FOR csvElm IN 1..(csvListTable.COUNT - 1) LOOP
        dbms_output.put_line('-- CSV element        : <'||csvListTable(csvElm)||'>');
        dbms_output.put_line('-- Trimmed CSV element: <'||trim(csvListTable(csvElm))||'>');
    END LOOP; 
END;
/

样本输出:

-- CSV element        : <elm1>;
-- Trimmed CSV element: <elm1>;
-- CSV element        : < elm2>;
-- Trimmed CSV element: <elm2>;
-- CSV element        : <elm3 >;
-- Trimmed CSV element: <elm3>;
-- CSV element        : <elm4 >;
-- Trimmed CSV element: <elm4>;
-- CSV element        : < elm5>;
-- Trimmed CSV element: <elm5>;

【讨论】:

【参考方案4】:

可以使用可在for 循环中使用的函数(ThinkJet 不带regexp):

创建类型和函数

CREATE OR REPLACE TYPE t_my_list AS TABLE OF VARCHAR2(100);
CREATE OR REPLACE
       FUNCTION cto_table(p_sep in Varchar2, p_list IN VARCHAR2)
         RETURN t_my_list
       AS
         l_string VARCHAR2(32767) := p_list || p_sep;
         l_sep_index PLS_INTEGER;
         l_index PLS_INTEGER := 1;
         l_tab t_my_list     := t_my_list();
       BEGIN
         LOOP
           l_sep_index := INSTR(l_string, p_sep, l_index);
           EXIT
         WHEN l_sep_index = 0;
           l_tab.EXTEND;
           l_tab(l_tab.COUNT) := TRIM(SUBSTR(l_string,l_index,l_sep_index - l_index));
           l_index            := l_sep_index + 1;
         END LOOP;
         RETURN l_tab;
       END cto_table;
/

那么如何在for循环中调用它:

DECLARE
  parm1 varchar2(4000) := '123,234,345,456,567,789,890';
BEGIN
    FOR x IN (select * from (table(cto_table(',', parm1)) ) )
    LOOP
        dbms_output.put_line('callProdcedure2 called with ' || x.COLUMN_VALUE);
        callProdcedure2(x.COLUMN_VALUE);
    END LOOP;
END;
/

注意 Oracle 给出的默认名称 COLUMN_VALUE,这是我想要使用结果所必需的。

结果如预期:

callProdcedure2 called with 123
callProdcedure2 called with 234
...

【讨论】:

以上是关于如何遍历 Oracle PLSQL 中的分隔列表的主要内容,如果未能解决你的问题,请参考以下文章

oracle中PLSQL存储过程中如何使用逗号分隔的集合(逗号分隔字符串转换为一个集合)

如何通过 Oracle 中的 regexp_replace 从逗号分隔列表中删除重复项?

如何通过 Oracle 中的正则表达式从逗号分隔列表中删除重复项,但我不想要重复值? [复制]

如何通过 Oracle regexp_replace 中的正则表达式从逗号分隔列表中删除重复项? [复制]

遍历并行列表并将一个列表中的制表符分隔文本转换为另一个列表中的 csv 路径

如何遍历 CloudFormation 模板中的值