메뉴 건너뛰기

조회 수 7704 추천 수 0 댓글 0
?

단축키

Prev이전 문서

Next다음 문서

크게 작게 위로 아래로 댓글로 가기 인쇄
?

단축키

Prev이전 문서

Next다음 문서

크게 작게 위로 아래로 댓글로 가기 인쇄
db 프로그래밍을 짜실 때 꼭 한가지를 기억하세요.. 저는 자바로 설명하겠지만 이건 꼭 언어를 따지는
건 아닙니다.

가령 이런 SQL문이 있다고 하죠.

select * from tab where id = 1;

그리고 시스템에서 id 를 입력받아서 매번 각 id에 대해 위의 쿼리를 실행한다고 하죠.

그러면 다음과 같이 쿼리를 생성해서는 안됩니다.

Connection con = null;
Statement stmt = null;

String query = "select * from tab where id = " + id;

stmt = con.createStatement(stmt);
ResultSet rs = stmt.executeQuery();

이와같이 코딩하면 SQL문장이 DBMS에 개별적으로 들어갑니다.
즉,
select * from tab where id = 1;
select * from tab where id = 2;
select * from tab where id = 3;
select * from tab where id = 4;
select * from tab where id = 5;

가 전부다 다른 문장으로 처리된다는 말이죠. 다른 문장으로 처리된다는 말은 매번 저 문장이
파싱되야한다는 말입니다.

그러나 쿼리를

select * from tab where id = ?

와 같이 만들고 ? 자리에 값을 넣는 방법이 어떤 언어든지 있습니다.

이럴때 ? 를 바인드 변수라고 하는데, 바인드 변수를 사용하면 위의쿼리는

select * from tab where id = :id_val;

이렇게 치환되죠..

그리고 개별 쿼리시마다 동일한 sql문을 수행하되 id_val의 값이 바뀌게 됩니다.

자바의 경우는 이것이 PreparedStatement입니다. 또 어떤분은 PreparedStatement를 쓰긴
하는데 정말 이상하게 씁니다. 가령 다음과 같은 식이죠.

String query = "select * from tab where id = " + id;
PreparedStatement pstmt = con.prepareStatement(query);

이런식이죠... 이렇게 해서야 바인드 변수가 쓰이지 않습니다.

String query = "select * from tab where id = ?"
PreparedStatement pstmt = con.prepareStatement(query);
pstmt.setInt(1, id);

이런식으로 코딩해야합니다..

그러면 DBMS에는 하나의 쿼리인 select * from tab where id = ? 만 들어가고
DBMS는 이 문장만 파싱한뒤에 바인드 변수값 바뀌는 처리는 파싱을 제외하고 해줄 수 있습니다..

파싱이 뭐 대단하겠냐 하겠지만 파싱이 많으면 CPU점유율이 높아집니다.. 매일 CPU한계치에
도달해서 뺑뺑이 치는 DBMS에서 이와같이 PreparedStatement를 쓰고 안쓰고의 차이는
심지어 CPU 차지율을 20% 이상 차이나게 합니다..

(말이 20% 지 평상시에 40% 의 CPU가 사용되는 시스템은 60% CPU를 먹게 만든단 뜻입니다..)

Statement로만 도배를 해놓은 경우 급하면 급한대로 오라클의 init{SID}.ora파일에서
CURSOR_SHARING=FORCE로 선언해주면 급하게 해결을 볼 수 있습니다..

그러나 오라클에서는 DSS나 QUERY REWRITE를 사용하는 환경에서는
이 매개변수를 FORCE로 고치지 말라고 경고하고 있습니다.

얼핏 생각해봐도 아시겠지만, CURSOR_SHARING=FORCE로 해주면 대부분의 쿼리를 오라클이 임의로 수정하여
바인드 변수를 쓰게 만들어 실행계획이 매번 동일하게 돌아가는 효과를 낳습니다.
따라서 QUERY REWRITE와 DSS에서 필요한 그때 그때에 맞는 정확한 플랜이 안나오게
될 수 있다는 것이죠.

QUERY REWRITE를 쓰는지 안쓰는지 모르겠다고 그러시면 안쓰는 것입니다. 제가 알기로는
QUERY REWRITE가 8i 이하에서는 Materialized View외에서는 사용되지 않습니다..
9i 부터는 모르겠군요.

만약 자신의 데이터베이스에서 얼마나 많은 재파싱이 일어나고 싶은지를 알고 싶다면,
다음 스크립트를 sys로 접근한뒤 실행해보세요.


prompt
prompt * SQL문 parsing time 구하기
prompt
SELECT         NAME,
        VALUE
FROM V$SYSSTAT
WHERE NAME = 'parse time cpu'
or NAME = 'parse time elapsed'
or NAME like 'parse count%';


결과는 대략 다음과 같이 나타납니다.


NAME                                  VALUE
------------------------------ ------------
parse time cpu                            0
parse time elapsed                        0
parse count (total)                 322,128
parse count (hard)                      662

SQL>


여기서 total 은 전체 파싱된 sql문장의 수 이며 (hard)는 하드 파싱, 즉 새로
파싱된 문장의 수를 의미합니다.

이 비율과 parse time cpu 에 나타난 parsing 에 소요된 cpu 시간을 사용해
접근하시면 됩니다.

List of Articles
번호 제목 날짜 조회 수
81 UNION ALL 과 ORDER BY 섞어서 쓰기 2016.08.30 9898
80 데이터베이스 백업하기 ( import : 가져오기 , export : 내보내기 ) file 2016.12.08 9882
79 오라클 백업 및 복구(Export, Import) file 2016.12.08 9758
78 오라클11g DB 생성 file 2016.12.08 9727
77 oracle db 백업(full) 2016.09.11 9662
76 조건절에 WHERE 1=1, WHERE 1=0 사용하여 쿼리 간편하게 사용하기 file 2016.08.29 9613
75 다른 버전의 오라클 import, export 하기 file 2016.09.12 9475
74 간단한 쿼리로 일련번호 쉽게 넣기 - ROWNUM 사용 file 2016.08.29 9472
73 단일행을 다중 row로 분리, 다중 row 결과를 단일행으로 연결하는 방법 2016.08.29 9386
72 PL/SQL에서 자바(Java) 클래스(Class), 함수 실행 방법 file 2016.12.08 9384
71 WHERE절에서 Null 값 비교하기 file 2016.08.29 9354
70 ORA-01652 128(으)로 테이블 공간 ***에서 임시 세그먼트를 확장할 수 없습니다. file 2016.12.08 9254
69 Oracle에서 Foreign Key가 걸려있는 컬럼값 업데이트(update) 하기 file 2016.08.29 9169
68 [Oracle|오라클] 한글 초성 검색 함수 2016.12.08 9087
67 oracle 특수문자 입력시 Substitution Variable 변수치환을 없애는 방법 file 2016.08.30 9025
66 ORA-39142 덤프 버전이 틀려서 넣지 못할 때 2016.12.08 8969
65 Oracle 에서 테이블과 리소스의 존재 여부를 알아보는 쿼리 file 2016.08.30 8949
64 오라클 12c (Oracle) 에서 테스트를 위해 샘플 스키마 설치하기 file 2016.08.30 8929
63 시퀀스 초기화 프로시저 2016.09.12 8924
62 sqlplus에서 Ampersand나 특수 문자를 갖는 데이타를 insert하는 방법 sqlplus 데이터 insert 입력값 요구 2016.12.08 8911
Board Pagination Prev 1 2 3 4 5 6 7 Next
/ 7

하단 정보를 입력할 수 있습니다

© k2s0o1d4e0s2i1g5n. All Rights Reserved