더북(TheBook)

성능 향상을 위한 다중 로우 처리

정적 SQL이든 동적 SQL이든 한 개 이상의 결과를 반환하는 SELECT문의 결과 집합을 받아 오기 위해 필요한 것은 세 가지인데, 바로 커서와 결과를 받아 오는 그릇 역할을 하는 커서나 레코드 변수, 그리고 LOOP나 FOR 같은 반복문이다. 즉 SQL문에 대한 커서를 연 뒤, 루프를 돌며 결과를 패치해 변수에 담고 루프가 끝나면 커서를 닫는다. 그런데 만약 쿼리 결과로 반환되는 로우 수가 많다면 루프를 돌면서 결과를 패치하는 식의 처리는 시간도 많이 걸리고 성능 면에서 좋지 않다. 일일이 한 로우씩 읽어 가며 결과를 받아올 것이 아니라 SQL의 장점인 집합적으로 처리하는 방법이 있다면 더 좋지 않을까? 바로 BULK COLLECT INTO 절을 사용해서 이를 구현할 수 있다. 먼저 정적 SQL을 이용한 예제를 살펴 보자.

입력

    -- BULK COLLECT INTO를 사용한 정적 SQL
    DECLARE
      -- 레코드 선언
      TYPE rec_physicist IS RECORD  (
        ids ch13_physicist.ids%TYPE,
        names ch13_physicist.names%TYPE,
        birth_dt ch13_physicist.birth_dt%TYPE );

      -- 레코드를 항목으로 하는 중첩 테이블 선언
      TYPE NT_physicist IS TABLE OF rec_physicist;

      -- 중첩 테이블 변수 선언
      vr_physicist NT_physicist;
    BEGIN
      BULK COLLECT INTO절(패치가 한 번에 이루어 진다)
      SELECT *
        BULK COLLECT INTO vr_physicist
        FROM ch13_physicist;
      -- 루프를 돌며 출력(이 루프는 값을 패치하는 것이 아니라 출력하기 위한 루프임)
      FOR i IN 1..vr_physicist.count
      LOOP
        DBMS_OUTPUT.PUT_LINE(vr_physicist(i).names);
      END LOOP;
    END;

결과

    Galileo Galilei
    Isaac Newton
    Max Plank
    Albert Einstein

쿼리를 수행한 결과 로우 수가 1개일 때는 SELECT INTO를, 2개 이상일 때는 커서를 사용해 결과를 변수에 담았는데, 위 예제에서는 SELECT … BULK COLLECT INTO 구문을 사용해 컬렉션 변수에 결과 집합을 담았다. 기존에 커서를 사용해 루프를 돌며 로우 하나씩 데이터를 받은 것에 비하면 코드도 훨씬 줄어 들었을 뿐만 아니라 성능 면에서도 우수하다. BULK COLLECT INTO 절은 동적 SQL에서도 사용할 수 있다. 정적 SQL에서는 SELECT 구문에 직접 들어갔지만, 동적 SQL에서는 EXECUTE IMMEDIATE 문에 포함된다.

입력

    DECLARE
      -- 레코드 선언
      TYPE rec_physicist IS RECORD  (
        ids ch13_physicist.ids%TYPE,
        names ch13_physicist.names%TYPE,
        birth_dt ch13_physicist.birth_dt%TYPE );
      -- 레코드를 항목으로 하는 중첩 테이블 선언
      TYPE NT_physicist IS TABLE OF rec_physicist;

      -- 중첩 테이블 변수 선언
      vr_physicist NT_physicist;

      vs_sql VARCHAR2(1000);
      vn_ids ch13_physicist.ids%TYPE := 1;
    BEGIN
      -- SELECT 구문
      vs_sql := 'SELECT * FROM ch13_physicist WHERE ids > :a' ;

      -- EXECUTE IMMEDIATE .. BULK COLLECT INTO 구문
      EXECUTE IMMEDIATE vs_sql BULK COLLECT INTO vr_physicist USING vn_ids;

      -- 루프를 돌며 출력
      FOR i IN 1..vr_physicist.count
      LOOP
        DBMS_OUTPUT.PUT_LINE(vr_physicist(i).names);
      END LOOP;

    END;

결과

    Isaac Newton
    Max Plank
    Albert Einstein

동적 SQL에서는 BULK COLLECT INTO 절이 EXECUTE IMMEDIATE 문의 쿼리 문자열과 USING 사이에 위치한다. 동적 SQL 역시 BULK COLLECT INTO 절을 사용하는 것이 커서 보다는 가독성과 성능 면에서 우수하므로, 결과 집합 건수가 많을 때는 이 방법을 사용하자.

신간 소식 구독하기
뉴스레터에 가입하시고 이메일로 신간 소식을 받아 보세요.