개발자가 흔히 실수하는 SQL 10가지 (2. 인덱스 스캔, 풀 스캔)
만일 총 직원수가 3만명인 사원테이블(BIG_EMP)에
메니져사원번호(MGR )가 7698 인 직원이 10,000 명이고,
부서번호(DEPTNO )가 60 인 직원이 1,200 명인 상황에서
SELECT PAYMENT FROM BIT_EMP
WHERE MGR='7698' AND DEPTNO='60';
위와 같은 쿼리에서 만일 메니져사원번호(MGR) 컬럼만 인덱스가 생성되어있다면
DB는 인덱스를 통하여 메니져사원번호가 '7698' 인 레코드를 검색후
해당 레코드의 부서번호가 60인지 판단하는 식으로
조회할 것이다.
하지만 이러한 인덱스컬럼의 조건절이 대부분의(15%이상) 레코드의 조건에 매칭되는 경우에서의
인덱스 스캔방식의 조회는 비효율적이다.
이유는 인덱스 스캔방식에서는 실제테이블 레코드를 조회할때 Single Block I/O 를 사용하기 때문이다.
차라리 인덱스컬럼을 무시하고 전체 테이블을 읽는 풀 스캔이 더 효율적이다.
풀스캔은 Multi Block I/O 를 사용한다.
Single Block I/O
인덱스를 사용하여 검색(인덱스 스캔)을 하게 되면 한번의 I/O 요청에 한 블록씩 데이터를 읽는다.
- 인덱스 테이블에서 조건절 검색 MGR='7698'
- ROWID 로 테이블 검색 (1만 ROW , 1만번 I/O)
- 조건절 검색 ( DEPTNO='60')
Multi Block I/O
전체 데이블 검색(풀 스캔)은 데이터를 읽을 때 한번의 I/O 요청으로 여러 블록을 한꺼번에 읽는다.
- 3만 / 150 = 2백 번 IO
- 조건절 검색 ( DEPTNO='SALES')
풀 스캔을 유도하는 방법
인덱스가 걸린 컬럼을 가공
SELECT SAL FROM BIG_EMP
WHERE DEPTNO='60' AND MGR || ‘’ ='7698‘;
힌트를 사용
SELECT /*+ FULL(BIG_EMP) */
SAL FROM BIG_EMP
WHERE DEPTNO='60' AND MGR='7698';