3

我正在使用存储过程来使用游标选择记录。此过程获取记录 ID 作为输入。

这是代码:

create or replace
procedure GET_ITEM_DETAILS_ALL
(
  cur_out out sys_refcursor,
  PSTRING VARCHAR2 DEFAULT NULL
)
is
  query_string  VARCHAR2(1000);
Begin
  query_string := 'Select IT.SL_NO from ITEM_DETAILS IT where IT.SL_NO in (:sl) order by IT.SL_NO';
  OPEN cur_out FOR query_string USING PSTRING;
End;

对于调用/测试程序,我使用以下代码:

VAR R REFCURSOR
EXEC GET_ITEM_DETAILS_ALL (:R, '4')
PRINT R 

问题是当我传递一个整数,如“4”或“2”时,程序返回数据,但当我传递“1,2”或“3,4”时,它显示错误。

4

4 回答 4

2
   OPEN cur_out FOR Select IT.SL_NO from ITEM_DETAILS IT where IT.SL_NO in (SELECT REGEXP_SUBSTR(sl,'[^,]+', 1, LEVEL) FROM DUAL CONNECT BY REGEXP_SUBSTR(sl, '[^,]+', 1, LEVEL) IS NOT NULL ) order by IT.SL_NO
于 2015-12-31T06:14:16.740 回答
0

您不能使用一个绑定变量将值列表传递给查询。
传递带有逗号分隔列表的字符串后,您的查询将变得
Select IT.SL_NO from ITEM_DETAILS IT where IT.SL_NO in ('1,2') order by IT.SL_NO
不正确。
您应该手动将输入字符串转换为查询中的值列表。

于 2013-03-03T09:12:16.957 回答
0

问题是 PSTRING 是单个变量而不是数组。所以你的陈述实际上等同于

.... where IT.SL_NO = PSTRING

这就是为什么它在您通过时有效,4而在您通过时失败1,2

以您的方式使用动态 SQL 没有任何价值(我们可以在不使用字符串的情况下打开引用游标)。但是,利用动态 SQL 是解决问题的一种方法:

query_string := 'Select IT.SL_NO from ITEM_DETAILS IT where IT.SL_NO in ('
                   ||PSTRING||
                   ') order by IT.SL_NO';
OPEN cur_out FOR query_string;

或者,您可以使用字符串标记器将字符串转换为标记。不幸的是,Oracle 没有内置标准,但针对不同版本的数据库有各种解决方法。Adrian Billington在他的网站上有一个很好的综述。使用其中一种方法可以让您放弃动态 SQL:

OPEN cur_out FOR select IT.SL_NO from ITEM_DETAILS IT 
   where IT.SL_NO in ( select * from table ( your_string_tokenizer( PSTRING ) ) ) 
   order by IT.SL_NO;
于 2013-03-03T09:27:53.170 回答
0

解决方案如下:

create or replace procedure GETITEM_DETAILS_ALL

(
    cur_out out sys_refcursor,
    PSTRING VARCHAR2 DEFAULT NULL
 )

is

query_string  VARCHAR2(1000);

Begin

query_string := 'Select IT.SL_NO from ITEM_DETAILS IT where IT.SL_NO in (' || PSTRING || ') order by IT.SL_NO';

OPEN cur_out FOR query_string;

End;
于 2013-03-03T12:15:38.290 回答