얼마전 고객사 DB를 튜닝하다가 생긴 일이다.
TEST_TAB 테이블에는 26만건의 데이터가 들어 있다.
SET STATISTICS IO ON
SET STATISTICS TIME ON
SELECT MAX(SEQ) + 1 FROM TEST_TAB
INDEX가 없을 때, 위의 SQL을 실행시켰을 때의 IO와 실행시간을 측정하면 다음과 같다.
(1개 행 적용됨)
Table 'TEST_TAB'. Scan count 1, logical reads 713, physical reads 0, read-ahead reads 0.
SQL Server Execution Times:
CPU time = 297 ms, elapsed time = 301 ms.
이 SQL이 바로 고객사 DB서버 CPU를 100%치게 만든 악성 SQL 세 개 중 하나였다.
물론, 실행되는 빈도가 높지 않다면 별 문제가 안되겠지만, 이런 SQL이 문제가 되는 것은 무조건 Table Full Scan을 하기 때문에 DATA가 쌓일 수록 영향이 커진다는 것이다.
해결책은 간단하다. SEQ 컬럼에 INDEX를 친 후 다시 한번 실행하면 다음과 같다.
Table 'TEST_TAB'. Scan count 1, logical reads 3, physical reads 0, read-ahead reads 0.
SQL Server Execution Times:
CPU time = 0 ms, elapsed time = 0 ms.
COUNT함수도 마찬가지다.
SELECT COUNT(*) FROM TEST_TAB WHERE MANAGE = 1
위의 SQL을 실행시켰을 때의 부하는 다음과 같다.
Table 'TEST_TAB'. Scan count 1, logical reads 713, physical reads 0, read-ahead reads 136.
SQL Server Execution Times:
CPU time = 63 ms, elapsed time = 143 ms.
MANAGE 컬럼에 인덱스를 친 후엔 다음과 같이 된다.
Table 'TEST_TAB'. Scan count 1, logical reads 3, physical reads 0, read-ahead reads 0.
SQL Server Execution Times:
CPU time = 0 ms, elapsed time = 0 ms.
결론은 이거다.
사용해야 할 경우, 반드시 인덱스를 고려하라
'SQL Server' 카테고리의 다른 글
대용량 로그테이블 삭제하기 (DELETE 문에 TOP 구현하기) (2) | 2008.08.23 |
---|---|
여러개의 테이블에서 JOIN하여 UPDATE 및 DELETE하는 SQL (5) | 2008.08.21 |
데드락이 발생할 때의 원인분석 (0) | 2008.08.17 |
INSERT문 실행 후 INSERT된 IDENTITY를 확인하는 방법 (1) | 2008.08.15 |
SQL Server의 SQL문에서 이벤트로그 기록하기 (0) | 2008.03.17 |