문제

IN 절을 제공하는 데 사용되는 가변 개수의 매개변수 값을 허용하는 Oracle 저장 프로시저를 어떻게 만들 수 있습니까?

이것이 내가 달성하려는 것입니다.업데이트하려는 행의 기본 키 변수 목록을 전달하기 위해 PLSQL에서 선언하는 방법을 모르겠습니다.

FUNCTION EXECUTE_UPDATE
  ( <parameter_list>
   value IN int)
  RETURN  int IS
BEGIN 
    [...other statements...]
    update table1 set col1 = col1 - value where id in (<parameter_list>) 

    RETURN SQL%ROWCOUNT ;
END;

또한 이 프로시저를 C#에서 호출하고 싶으므로 .NET 기능과 호환되어야 합니다.

고마워요, 로버트

도움이 되었습니까?

해결책

요소 자체에 문자열이 포함되지 않을 것이라고 100% 확신할 수 있다고 가정하면 CSV를 사용하는 것이 아마도 가장 간단한 방법일 것입니다.

이를 수행하는 대안이자 더 강력한 방법은 문자열 테이블로 사용자 정의 유형을 만드는 것입니다.문자열이 100자를 넘지 않는다고 가정하면 다음과 같은 결과를 얻을 수 있습니다.

CREATE TYPE string_table AS TABLE OF varchar2(100);

그런 다음 이 유형의 변수를 저장 프로시저에 전달하고 직접 참조할 수 있습니다.귀하의 경우에는 다음과 같습니다.

FUNCTION EXECUTE_UPDATE(
    identifierList string_table,
    value int)
RETURN int
IS
BEGIN

    [...other stuff...]

    update table1 set col1 = col1 - value 
    where id in (select column_value from table(identifierList));

    RETURN SQL%ROWCOUNT;

END

그만큼 table() 함수는 사용자 정의 유형을 단일 열 "COLUMN_VALUE"가 있는 테이블로 변환합니다. 그런 다음 이를 다른 테이블처럼 처리할 수 있습니다(조인 또는 이 경우 하위 선택도 마찬가지입니다).

이것의 장점은 Oracle이 사용자를 위해 생성자를 생성하므로 저장 프로시저를 호출할 때 간단히 다음과 같이 작성할 수 있다는 것입니다.

execute_update(string_table('foo','bar','baz'), 32);

저는 여러분이 C#에서 프로그래밍 방식으로 이 명령을 작성하는 작업을 처리할 수 있다고 가정합니다.

여담으로, 우리 회사에는 문자열, double, int 등의 목록에 대한 표준으로 정의된 이러한 사용자 정의 유형이 많이 있습니다.우리는 또한 활용합니다 Oracle JPublisher 이러한 유형에서 해당 Java 객체로 직접 매핑할 수 있습니다.주위를 둘러보았지만 C#에 직접적으로 상응하는 항목을 찾을 수 없었습니다.Java 개발자가 이 질문을 접할 경우를 대비해 언급하겠다고 생각했습니다.

다른 팁

Mark A. Williams의 다음 기사를 찾았습니다. 이 기사는 연관 배열을 사용하여 C#에서 PL/SQL Procs로 배열을 전달하는 좋은 예를 제공합니다 (유형 MyType는 mytable의 테이블입니다.

Mark A. Williams의 훌륭한 기사

가변 수의 매개 변수로 프로 시저를 만들 수있는 직접적인 방법이 없다고 생각합니다. 그러나 문제에 대한 적어도 부분적인 솔루션이 있습니다. 여기.

  1. 일반적인 통화 유형 절차가있는 경우 과부하가 도움이 될 수 있습니다.
  2. 매개 변수 수에 상한이 있고 (그리고 그 유형이 미리 알려져 있음), 매개 변수의 기본값이 도움이 될 수 있습니다.
  3. 가장 좋은 옵션은 데이터베이스 커서에 대한 포인터가 무엇인지 커서 변수의 사용 일 수 있습니다.

불행히도 .NET 환경에 대한 경험이 없습니다.

긴 매개 변수 목록을 사용하고 값을 테이블 생성자에로드하지 않겠습니까? 이 트릭에 대한 SQL/PSM은 다음과 같습니다

UPDATE Foobar
   SET x = 42
 WHERE Foobar.keycol
      IN (SELECT X.parm
            FROM (VALUES (in_p01), (in_p02), .., (in_p99)) X(parm)
           WHERE X.parm IS NOT NULL);

왜 저장된 절차 여야합니까? 프로그래밍 방식으로 준비된 진술을 구축 할 수 있습니다.

Oracle에 대해서는 수행하지 않았지만 SQL Server를 사용하면 CSV 문자열을 테이블로 변환하여 IN 절에 사용할 수 있습니다. Oracle을 위해 이것을 다시 작성하는 것은 간단해야합니다 (나는 생각합니다!).

CREATE Function dbo.CsvToInt ( @Array varchar(1000)) 
returns @IntTable table 
    (IntValue nvarchar(100))
AS
begin

    declare @separator char(1)
    set @separator = ','

    declare @separator_position int 
    declare @array_value varchar(1000) 

    set @array = @array + ','

    while patindex('%,%' , @array) <> 0 
    begin

      select @separator_position =  patindex('%,%' , @array)
      select @array_value = left(@array, @separator_position - 1)

        Insert @IntTable
        Values (Cast(@array_value as nvarchar))

      select @array = stuff(@array, 1, @separator_position, '')
    end

    return
end

그런 다음 CSV 문자열 (예 : '0001,0002,0003')을 전달하고

UPDATE table1 SET 
       col1 = col1 - value 
WHERE id in (SELECT * FROM csvToInt(@myParam)) 

AskTom 웹 사이트에는 CSV 문자열을 구문 분석하고 주어진 SQL Server 예제와 유사한 방식으로 문서에서이를 사용하는 기능을 작성하는 방법을 보여주는 기사가 있습니다.

보다 Asktom

라이센스 : CC-BY-SA ~와 함께 속성
제휴하지 않습니다 StackOverflow
scroll top