2008년 11월 14일 금요일

ORA-1555 (SNAPSHOT TOO OLD)의 일반적인 원인 및 조치사항

ORA-1555 (SNAPSHOT TOO OLD)의 일반적인 원인 및 조치사항
===========================================

PURPOSE
-------

ORA-1555 (snapshot too old)는 db 관리 업무에 익숙하지 않은 경우, rollback
관련된 오류 중 혼란을 일으키기 쉬운 오류이다.
이미 문서 <bulletin:11152>와 그외 자료가 이 오류를 설명하고 해결하기 위해
만들어져 있지만, ORA-1555 원인 파악을 위해 내용이 다소 길고 복잡하게
구성되어 있는 편이다.
여기에서는 발생 가능한 여러가지 원인 중 일반적인 원을을 위주로, 초보자도
쉽게 이해할 수 있도록 간단히 설명한다.


Explanation
-----------

일반적으로 ORA-1555에 혼란을 일으키는 원인은 한편으로는 오류 메시지 자체에
있다고 볼 수 있다.

ORA-1555: snapshot too old: rollback segment %s too small

이와 같은 오류에서 마치 ora-1555가 rollback segment에 write시 space가
부족해서 발생하는것으로 착각하는 사용자가 많다.

중요한 것은 ORA-1555는 rollback segment에 정보를 write시에 발생하는 것이
아니고 rollback segment로 부터 before image를 읽으려는 시점에서 발생한다는
것이다.

쉬운 예를 들어보자.

(1) 사원이 천명인 회사에서 select한 문장으로 그 전체 사원의 정보를 읽는데
10분이 걸린다고 가정한다.
(2) 100번 사원 정보를 읽는데, 아직 읽지 않은 700번 사원에 대해 다른 session에서
급여를 인상하는 update문장을 수행하고 commit을 한다.
select문장은 lock을 걸지 않기 때문에 select도중 다른 update문장이
수행되고 commit하는데 아무 문제가 없다.
(3) 1번에서 수행중인 select문장이 계속 진행되면서 700번 사원 정보를 읽으려고
하면 이 정보가 수정되어 변경되었음을 알게 된다.
그럼 select문장은 정보의 일관성을 위해 첫번째 사원을 읽기 시작한 시점의
700번 사원에 대한 정보를 읽기 위해, 즉 before image를 읽기 위해
rollback segment를 찾아간다.
(4) rollback segment내에 급여 인상 전 정보가 있으면 읽는다.
단 이때,
이 시스템에 트랜잭션이 매우 많아서 commit이 매우 많이 발생한 경우
이미 2번에서 변경하고 commit한 정보는 다른 트랜잭션에서 overwrite했을
수 있다.
이런 경우 before image를 읽으러 간 select문장은 ora-1555를 만나게 되는
것이다.
(5) 4번에서 ora-1555를 만난 경우 다시 동일한 select문장을 수행하면,
이번에는 이미 급여가 인상된 후의 시점에서 시작하므로 700번 사원을
읽는 경우에도 급여 인상전의 before image가 필요하지 않아 ora-1555는
다시 발생하지 않을 수 있다.
이러한 이유로 ora-1555는 발생했다 안했다 하는 식으로 일정하게 발생되지
않고, 조치 방법이라는것도 100% 안전하기보다는 확률적으로 충분히 만나지
않을 수 있는 환경을 만드는것이라고 볼 수 있다.

결국 ora-1555가 발생하는 것은 읽어야 하는 before image가 다른 트랜잭션에
의해 이미 overwrite되어 읽을 수 없는 경우 발생하므로, 발생하지 않게 하기
위해서는 데이타를 조회시 consistency를 유지해야 하는 시점동안 가능하면
오래 동안 rollback의 image가 유지되어야 하는것이다.

이렇게 이미 기록된 정보를 가능하면 오랜 기간동안 유지한다는 것은 새로운
트랜잭션의 기록을 위해 space를 확보해야 하는 작업과는 반대된다.
즉, ORA-1562와 같이 rollback segment를 write시에 space가 부족하여
space를 확보하기 위한 조치 방법과, 이 ORA-1555의 조치 방법을 서로 상충되어
trade-off가 있음을 주의해야 한다.

두 오류를 모두 피해가기 위해서는 일반적으로 매우 큰 rollback space가
도움이 된다.

ORA-1555의 일반적인 발생 경우 및 해결 방법을 정리한다.

(1) 트랜잭션에 비해 rollback segment 갯수가 적은 경우

rollback segment하나에 동시에 기록 가능한 트랜잭션의 수는 rollback
segment header내의 transaction table의 entry갯수로 제한되어 있다.
이 수는 oracle version마다 다르지만 8i이상부터는 약 20개 정도이다.
(transactions_per_rollback_segment의 지정과는 무관한다.)
기본적으로 install시 생성되는 rollback segment는 4개인데, 이대로 놓고
사용한다면, 결국 80 (20 * 4) 만큼의 commit이 발생하고 난 뒤에는
다시 처음부터 transaction table의 entry 중 commit된 트랜잭션의
정보를 가지는 entry의 정보를 overwrite하게 되는 것이다.

해결 방법: rollback segment갯수를 증가시킨다.
즉 새로운 rollback segment를 create시킨다.

부작용: 제한된 rollback tablespace공간 내에서, 여러개의 rollback
segment를 유지하는것은 하나의 rollback segment가 평균 가질 수
있는 space가 그만큼 줄어드는 셈이다.
이 부작용까지 줄이려면, rollback tablespace자체가 충분히
커야 하고 space를 많이 요구하는 트랜잭션은 'set transaction
use rollback segment' 문장을 이용하여 큰 rollback을 지정하여
사용하도록 한다.

(2) rollback segment를 shrink하거나 optimal이 설정된 경우

rollback segment를 shrink하거나 optimal을 지정하게 되면 이미 쓰여진
rollback의 before image를 다른 트랜잭션이 overwrite도 하기 전에 미리
지워 버리게 되는 셈이다.
그러므로 이런 경우도 ora-1555의 원인이 된다.

해결 방법: optimal을 너무 적게 지정하지 말고, shrink를 너무 자주
하지 않는다. shrink를 수행 후 ora-1555가 발생하는 경우,
단지 다시 조회하는것만으로 앞의 예제 (5)번에서 설명한
이유로 인해, 해결되는 경우가 많다.

(3) proc와 같은 application에서 loop내의 fetch문장에서 자주 commit을
하는 경우

fetch문장은 loop를 도는 동안 일정하게 read consistency를 유지해야 한다.
그리고 미리 cursor를 정의시에 데이타를 읽어두는것이 아니고, fetch시에
loop를 돌면서 그때그때 데이타를 읽게 된다.
그런데 loop내의 dml에 대해 너무 자주 commit을 하게 되면 그만큼
여러개의 트랜잭션이 처리된 결과로 rollback segment의 transaction table이
빨리 사용되고 overwrite되게 된다.

해결 방법: loop내의 commit횟수를 줄인다. 예를 들어 loop를 돌때마다
commit하게 하였다면 천번에 한번 혹은 만번 loop를 돈 후
commit하는 식으로 늘려준다.

이 외에도 rollback tablespace자체의 space가 부족하여 transaction table의
entry들이 아직 overwrite되지도 않았는데, commit된 transaction이 사용한
rollback segment내의 space가 먼저 overwrite되는 경우도 있다.
그러나 일반적으로 rollback segment의 space를 너무 작게 유지하지는 않기
때문에 이렇게 space부족으로 ora-1555를 만나는 경우는 많지 않다.
이렇게 space가 절대적으로 부족한 경우는 rollback에 write하는 시점에서,
ora-1562가 먼저 발생하게 된다.

ora-1562에 대해서는 <bulletin:10823> "ORA-1562 분석 및 해결 방법
(ROLLBACK SEGMENT 크기 문제)"를 참조하고,
좀더 자세한 ora-1555의 개념에 대해서는 <bulletin:11152> "ORA-1555 원인
분석 및 조치 사항" 을 참조한다.


http://kr.forums.oracle.com/forums/thread.jspa?messageID=1698897


ORA-1555 조치방법 
게시일: 2007. 2. 4 오후 7:54
제품 : ORACLE SERVER

작성날짜 : 2005-11-06

ORA-1555 Snapshot Too Old 에러가 발생하는 원인은 여러가지가 있을 수 있다.
이를 이해하기 위해서 오라클의 내부 메카니즘에 대해서 알아볼 필요가 있다.
오라클은 항상 Statement-Level Read Consistency를 유지한다. 즉, 하나의
Query 에서 읽어들이는 값은 그 Query가 발생한 시점의 값, 즉 Snapshot 을
반영한다. 따라서 Query 가 진행되는 동안 데이타가 변하더라도 Query의
결과값에는 아무런 영향을 끼치지 않는다.

오라클은 특정한 시점을 SCN(System Change Number)로 관리한다. 달리 말하면
SCN은 특정한 시점에서의 데이타베이스의 상태라고도 할 수 있다. Read
Consistency를 유지하기 위하여 Query가 실행되는 순간 해당 시점의 SCN이
세팅된다. 어떠한 Query든 조회 시점 이후의 SCN 값을 갖는 데이타를 읽어들여
서는 안된다.

Read Consistent Snapshot을 구현하기 위한 방법으로 롤백 세그먼트가
사용된다. 데이타가 변경되면 변경되기 이전의 값은 롤백 세그먼트에 기록되고
데이타 블럭 헤더에는 변경되기 이전 값이 기록된 롤백 세그먼트 블럭의 위치가
기록된다. 모든 데이타 블럭에는 가장 최근의 커밋 시점의 SCN이 기록되어
있다. Query 가 진행되면 데이타 블럭에서는 Query의 SCN 보다 이전의 SCN값을
갖는 블럭만이 읽혀지고 그 이후의 SCN 값을 갖는 블럭과 커밋되지 않은 값을
갖는 블럭에 대한 Query는 롤백 세그먼트에서 읽혀지게 된다. 이 때 만약 롤백
세그먼트에 원하는 블럭이 없어서 읽어들이지 못하게 되면 ORA-1555 에러가
발생하게 된다.

롤백 세그먼트는 트랜잭션이 끝나기 전까지 변경된 데이타의 원래 상태를
기록하고 있다. 그런데 트랜잭션이 종료되면 롤백 세그먼트에 의해 점유된
영역은 Free 상태로 되면서 다른 트랜잭션에 의해서 덮어씌워질 수도 있게
된다. 따라서 Query가 필요로하는 값을 가진 롤백 세그먼트 블럭이 더 이상
존재하지 않는 상황에서 그 블럭을 요구하게 되면 ORA-1555 에러가 발생하게
되는 것이다.

? ORA-1555 에러가 발생하는 경우

1. 데이타의 변경이 심한 데이타베이스에서 롤백 세그먼트의 갯수와 크기가
작을 경우에 발생한다. 많은 트랜잭션이 데이타를 자주 변경하고 커밋하게 되면
커밋된 트랜잭션이 이용하던 롤백 세그먼트 공간을 다른 트랜잭션이 이용하게
될 가능성이 많아진다.
따라서 긴 Query의 경우 원하는 값을 롤백 세그먼트에서 얻고자 할 때 이미
다른 트랜잭션이 그 값이 저장된 공간을 이용해 버리는 결과가 발생할 수
있다. 이와 같은 경우에는 크기가 큰 롤백 세그먼트를 이용하면 어느 정도 예방이
가능하다.

2. 롤백 세그먼트가 손상되어 읽을 수 없게 된 경우

3. Fetch Across Commit

한 테이블에 대하여 Query가 커서를 열고 루프 내에서 데이타를 Fetch하고
변경하고 커밋하는 과정에서 발생한다. 이 경우에는 ORA-1555 에러가 자주
발생할 소지가 있는데 예를 들어 커서가 SCN=10 에 Open되었다고 하자. 따라서
이 커서에 관련된 Fetch는 SCN<=10 인 블럭만을 읽어들여야 한다. 이
프로그램은 데이타를 Fetch 한 후에 변경하고 다시 커밋하는 과정을 계속
반복하는데 SCN=20에서 커밋했다고 하자. 만약 이후의 Query가 이전의 커밋된
블럭의 데이타를 요구할 경우 그 값이 이미 변경되었으므로 롤백 세그먼트를
검색 하지만 많은 변경이 있어왔기 때문에 SCN=10인 블럭을 찾지 못하고
ORA-1555 에러를 유발할 수 있다. 이를 방지하기 위해서는 커서가 Open된
상태에서는 커밋을 자주하지 않고 롤백 세그먼트 크기를 키워 나가도록 한다.

4. Fetch Across Commit 와 Delayed Block Clean Out

Delayed Block Clean Out 도 이 에러를 유발할 수 있다. 데이타 블럭이
변경되고 커밋되면 오라클은 롤백 세그먼트 헤더에 그 트랜잭션이 커밋되었다
고 기록하지만 데이타 블럭을 바로 변경하지는 않는다 (Fast Commit). 그리고
다음 트랜잭션이 변경된 블럭을 요구할 때야 비로소 변경 시키는데 이를
Delayed Block Clean Out 이라고 한다.
예를 들어 3 과 같은 경우에서 두개의 테이블에 대한 Query를 알아보자.
즉, 하나의 테이블로부터 데이타를 Fetch 하고 다른 테이블의 데이타를
변경한다고 하자. 데이타가 한쪽 테이블에서 커밋되고 있지만 데이타를
가져오는 테이블에 대해서는 Clean Out이 이루어지지 않았기 때문에 ORA-1555
에러가 발생할 수 있다. 이 문제는 커서를 사용하기 전에 Full Table Scan을
해주면 예방이 가능하다.
커서가 오픈된 상태에서의 커밋은 ANSI 표준에는 들어있지 않지만 오라클
에서는 지원이 된다. 하지만 ORA-1555 에러를 일으킬 수도 있다는 점에 유의
하여야 한다.


http://kr.forums.oracle.com/forums/thread.jspa?threadID=471354&tstart=285 

query시 발생하는 ORA-1555의 발생 원인과 조치 사항에 대해서 자세히 살펴
본다.

Explanation



ORA-1555가 발생하는 원인은 여러가지가 있지만 기본적으로는 사용자가 필요로 
하는 롤백 세그먼트의 정보가 다른 트랜잭션에 의해 overwrite되어, 존재하지 
않을 때 발생한다.

이 문서를 읽기 전에 기본적으로 알아야 하는 오라클의 read consistency와 
관련된 다음 내용들은 이 문서의 마지막에 별첨으로 용어 및 개념에 대해 설명
하였으므로 참고할 수 있다.

(1) SCN (System Change Number)
(2) statement-read level read consistent
(3) read consistent snapshot
(4) rollback segment의 wrap around/overwrite

ORA-1555에 관한 자세한 설명에 앞서, 데이타 블럭과 롤백 세그먼트 사이의 
구조에 대해 간단히 알아보도록 한다. 데이타 블럭의 헤더에는, 이 블럭 내에 
포함된 데이타를 변경한 트랜잭션의 정보와, 롤백 세그먼트 내의 해당 active
transaction을 가리키는 영역이 존재한다. 롤백 세그먼트는 세그먼트의 첫 번째
블럭을 헤더 블럭으로 사용하는데, 그 안에 이 롤백 세그먼트를 최근에 
사용한 트랜잭션들의 정보와, undo record들이 저장되어 있는 롤백 세그먼트 내의
주소가 저장되어 있는 트랜잭션 테이블이 포함되어 있다.
다음 예의 그림을 통해 다음과 같은 사항을 알 수 있다.

(1) 데이타 블럭 500번지의 row 2를 변경한 xid1 트랜잭션은 아직 commit
되지 않은 상태이다. 블럭의 헤더에는 트랜잭션이 아직 commit되지 
않았다는 정보와 5번 롤백 세그먼트 헤더 내의 3번째 엔트리에 트랜
잭션의 정보와, undo record를 얻을 수 있는 자세한 정보가 있음을 
알려준다.
(2) 롤백 세그먼트 5번의 3번째 슬롯은 이 트랜잭션이 변경한 undo record가 
롤백 세그먼트내의 7109번지에 저장되어 있음을 나타낸다. 2, 4, nn번 
엔트리의 경우는 이미 트랜잭션이 commit되었으므로, 다른 트랜잭션이 
이 엔트리를 overwrite할 수 있다.
(3) xid1 트랜잭션에 의해 변경된 undo record가 포함되어 있는 6900, 7109
블럭은 link로 연결되어 있어 xid1 트랜잭션이 변경한 모든 record들의 
before image를 구성할 수 있다.

Data Block 500 Rollback Segment Header 5
+------+-----------+---+ +----+------+----------+------+
txuncommitted5,3----+01xid4ACTIVE9012

+------+-----------+---+ | | 02 | xid9 | COMMITTED| 8100 |
| row 1 | --->| 03 | xid1 | ACTIVE | 7109 |---
| row 2 changed | | 04 | xid2 | COMMITTED| 7632 | |
... .. ........5098 
row n nnxidmCOMMITTED6777 

---------------------- +----+------+----------+------+ |

Block 6900 Block 7109 |
------------- ------------- |
| xid1 |<----+ | xid1 |<----+
| | | | | 
undo record  undo record
    
 +-------6900

------------- -------------
rollback segment block rollback segment block

ORA-1555가 발생하는 주요 원인과, 이 오류 발생을 최소화할 수 있는 방법은 
다음과 같다.

1. 데이타베이스에 변경을 가하는 트랜잭션은 많고, 롤백 세그먼트는 크기도
작고, 갯수도 적은 경우

다음과 같은 상황을 가정할 수 있다.
(1) 약 30분이 걸려서 A 테이블의 대부분을 읽어야 하는 긴 query 하나를 
수행시켰다.
이 때의 SCN이 10이었다.
(2) 위의 query가 결과값을 찾고 있는 동안, xid1 트랜잭션은 A 테이블에 
대해서 update작업을 수행하고 commit하여 A table이 저장되어 있는 
블럭 중 하나인 500번지 블럭의 SCN이 20으로 변경되었다 
(3) query가 진행중인 동안 매우 많은 트랜잭션들이 database를 변경하고 
commit하였다.
(4) 이 query가 500번지 블럭을 읽고자 할 때 SCN이 20임을 확인하고, 
xid1 트랜잭션에 의해 변경된 undo record를 찾기 위해 롤백 세그먼트를 
참조하였다.
(5) 그러나 xid1 트랜잭션은 이미 commit된 상태이고, query가 진행되는 
동안 매우 많은 트랜잭션이 데이타베이스 변경 작업을 수행한 결과 
롤백 세그먼트내의 xid1 트랜잭션의 undo record가 저장되어 있는
블럭이 다른 트랜잭션들에 의해 overwrite된 상태였다.
(6) ORA-1555가 발생한다.

해결 방법: 
(1) 롤백 세그먼트의 크기를 크게 하고 갯수를 늘리면, 롤백 세그먼트가
wrap around/overwrite되는 주기가 늦추어진다.
(2) 트랜잭션의 수행이 많은 때에는 수행 시간이 오래 걸리는 query 문은 
수행시키지 않도록 한다.

2. fetch across commit

프로그램내에서 cursor를 선언하고 loop를 수행하면서 fetch하고 데이타를 
변경하는 경우 많은 프로그래머들은 롤백 세그먼트의 사용량을 줄이기 위해서
매 loop시마다 commit을 한다. 그러나 cursor의 loop내에서 commit하는 
것은 ANSI standard에서는 제공하는 것이 아니며, ORA-1555를 발생시킬 가능
성이 있다.
ORA-1555가 발생하는 경우는 (1)의 경우와 유사하다. cursor는 선언하고, 
open시에 데이타를 읽는 것이 아니고 fetch 때마다 읽게 되므로 fetch를 
수행하는 것은 long query를 시작하는 것과 같다. 즉, fetch문의 loop를 
수행하는 동안, 처음 fetch문 수행시점의 SCN보다 작거나 같은 SCN의 데이
타를 읽어야 한다. 그런데 loop 수행시마다 데이타를 변경하고 commit하게
되면, commit한 block의 SCN은 증가되고 변경된 정보도 다른 트랜잭션에
의해 재사용되어질 수 있다. 이렇게 블럭은 변경되었으나, 변경된 정보가
이미 다른 트랜잭션에 의해 overwrite된 블럭의 데이타를 fetch하고자 
하면, 오라클은 read consistent snapshot을 구성할 수 없게 되므로 
ORA-1555가 발생하게 된다. 

해결 방법: 
(1) cursor 내에서 commit하는 횟수를 줄인다. 예를 들어 첨자를 이용해 
5만건에 한번씩 commit할 수 있으며, 이렇게 되면 5만건의 데이타를 
저장할 수 있는 큰 롤백 세그먼트가 있어야 한다.
(2) cursor 선언 시 구성될 active set의 범위를 줄인다. 즉 한번에 모든
데이타를 읽어 처리하기 보다는, where절을 이용하여 데이타를 나누어, 
여러번에 걸쳐 수행한다. 
(3) 1번의 경우와 마찬가지로, commit된 정보가 overwrite되는 주기를 
늦추기 위해서 롤백 세그먼트의 갯수를 증가시키고 그 크기도 크게하면 
도움이 된다.

3. delayed block clean out

오라클은 기본적으로 transaction이 commit하면, fast commit을 수행한다. 
즉, 트랜잭션이 데이타를 변경시키고 commit하면, 변경된 데이타 블럭의 
header부분에 트랜잭션이 commit되었음을 기록하는 것이 아니고 일단 롤백 
세그먼트의 헤더부분에만 commit되었음을 기록한다. 이 후 그 데이타 블럭을
다른 트랜잭션이 access하게 되면, 그때 롤백 세그먼트의 정보를 이용하여
데이타 블럭에 commit된 상태를 반영하여 clean out시키는 것을 delayed 
block clean out이라고 한다.

이 delayed block clean out이 어떻게 ORA-1555를 발생하게 되는지 다음의 
상황을 살펴보면 된다.

(1) 다음과 같은 초기 상태를 가정할 수 있다. 
500번지 데이타 블럭의 데이타를 변경하는 트랜잭션은 존재하지 않고, 
rollback segment 5번 header의 3, 4, nn번째 트랜잭션 엔트리는 다른 
트랜잭션에 의해 재사용되어 질 수 있다.

Data Block 500 Rollback Segment Header 5
+-------+-------------+ +----+------+-------------------+

txnone 01xid4ACTIVE

+-------+-------------+ | 02 | xid9 | ACTIVE |
row 1 03xid7COMMITTED
row 2 04xid2COMMITTED
... .. ........
row n nnxidmCOMMITTED

--------------------- +----+------+-------------------+

(2) xid1 트랜잭션이 update문을 이용하여 500번지 데이타 블럭의 2번째 
데이타를 변경하였다. 
500번지 데이타 블럭의 헤더에는 xid1 트랜잭션의 정보가 저장되고, 롤백 
세그먼트 5번의 트랜잭션 슬롯 3 (5,3)을 가리키게 된다. COMMITTED로 표시
되었던 트랜잭션 슬롯 3번은 이제 ACTIVE 상태로 변경되었다.

Data Block 500 Rollback Segment Header 5
+-------+-------------------+ +----+------+-------------------+

xid15.3uncommitted----+01xid4ACTIVE

+-------+-------------------+ | | 02 | xid9 | ACTIVE |
row 1+--->03xid1ACTIVE
row 2 changed 04xid2COMMITTED
... .. ........
row n nnxidmCOMMITTED

--------------------------- +----+-------+------------------+

(3) xid1 트랜잭션이 commit을 수행하였다.
오라클은 롤백 세그먼트 헤더의 트랜잭션 테이블에서 xid1 트랜잭션의 정보를
찾아서 commit되었다고 기록하였다. 그러나 500번지 블럭의 헤더에는 commit
되었다는 정보를 기록하지 않는다. (fast commit)

Data Block 500 Rollback Segment Header 5
+-------+-------------------+ +----+-------+------------------+
xid15.3uncommitted----+01xid4ACTIVE

+-------+-------------------+ | | 02 | xid9 | ACTIVE |
row 1+--->03xid1COMMITTED
row 2 changed 04xid2COMMITTED
... .. ........
row n nnxidmmCOMMITTED

--------------------------- +----+-------+------------------+

(4) 데이타베이스에 변경을 가하는 매우 많은 트랜잭션이 수행되었다.
매우 많은 트랜잭션이 수행되어 롤백 세그먼트 헤더내에 있는 트랜잭션 테이블
의 엔트리가 대부분 재사용되었다. 트랜잭션 xid50이 롤백 세그먼트 5번의 
3번째 슬롯이 COMMITTED로 표시되어 있으므로, 비어있는 엔트리로 인식하여 
xid50에 관한 정보를 저장하였다.

Data Block 700 Rollback Segment Header 5
+-------+-------------------+ +----+-------+------------------+

xid505.3uncommitted----+01xid31COMMITTED

+-------+-------------------+ | | 02 | xid46 | ACTIVE |
row 1 changed+--->03xid50ACTIVE
row 2 04xid60COMMITTED
... .. .......
row n nnxidmmACTIVE

--------------------------- +----+-------+------------------+

(5) 다른 트랜잭션이 데이타 블럭 500번지를 방문하였다.
새로운 트랜잭션인 xid70 트랜잭션이 500번지 블럭을 읽고자 하였다. (3)번의 
그림에서 보듯이, 500번지 블럭 헤더에는 아직 commit되지 않은 트랜잭션이 
이 블럭을 변경하였으며, before image를 구성할 수 있는 정보가 롤백 세그먼트 
5번, 엔트리 3번에 있음을 나타낸다. 그러나 5번 롤백 세그먼트 헤더 내에 있는 
트랜잭션 테이블의 3번 슬롯은 xid1번이 아닌 xid50번의 정보가 저장되어 있다. 
즉, delayed block cleanout이 이루어지기 전에 롤백 세그먼트 헤더가 overwrite
된 것이다.

(6) xid7 트랜잭션은 read consistent snapshot을 구성할 수 없으므로 
ORA-1555가 발생한다. 

해결 방법: 
(1) ORA-1555를 발생시킬 상황 이전에 읽고자 하는 테이블에 대해 full 
scan을 실시한다면, 롤백 세그먼트안의 정보가 overwrite되기 전에 
delayed block cleanout이 이루어지도록 할 수 있다.

(2) 1 ~ 4번의 모든 원인에 대해서 롤백 세그먼트를 크게 유지하면, 롤백 
세그먼트의 정보가 overwrite되는 주기를 늦출 수 있어 ORA-1555를 
피하는 데 도움이 될 수 있다.

4. OPTIMAL 크기가 아주 작을 때

롤백 세그먼트는 트랜잭션의 사용에 의해 한번 크기가 늘어나면 기본적으로 
그 롤백 세그먼트를 지우고 다시 만들기까지는 크기가 줄어들지 않는다. 
그러나 optimal size를 지정하게 되면, 롤백 세그먼트에서 새로운 extent를
요구하는 시점에, 현재 할당된 롤백 세그먼트의 크기와 optimal에 지정된 
크기를 비교하게 된다. 할당된 공간이 optimal 크기보다 큰 경우, 할당된 
extent중 active한 트랜잭션이 사용하고 있지 않은 extent들은 release시켜, 
롤백 테이블스페이스의 공간으로 환원된다.
그러므로 이 optimal size가 지나치게 작다면, 트랜잭션이 commit되자마자
롤백 세그먼트 내의 정보는 잃게 될 것이다. 그러나, 위의 1 ~ 4번에서 살펴보
았듯이 이미 commit된 트랜잭션의 정보라 하더라도 이후에 필요하게 되는 
경우가 발생하므로 이렇게 빈번히 commit된 트랜잭션의 정보가 포함되어 있는
롤백 세그먼트의 extent를 release시키는 것은 바람직하지 않을 수 있다.

해결 방법: 
(1) optimal을 지정할 때는 20개의 extents정도의 크기정도로 지정하는 
것이 적당하며, 그것보다 더 작게 지정하지 않도록 한다.
(2) 롤백 세그먼트를 많이 필요로 하는 batch job의 경우 set transaction
use rollback segment rollback_segment_name; 구문을 이용하여 특정 
롤백 세그먼트를 사용하게 하고 나머지 롤백 세그먼트들은 OLTP job이 
사용하도록 한다. 이렇게 하면 OPTIMAL을 지정하지 않아도 모든 롤백 
세그먼트가 불필요하게 확장되는 일을 막을 수 있다.

별첨: 용어 및 기본 개념 설명--------------------------------------------------------
(1) SCN(System Change Number)
오라클은 특정한 시점의 데이타베이스 상태를 SCN으로 관리한다. 트랜잭션이
commit되면, SCN은 최근의 SCN 보다 크고 유일한 값이 할당되며, 이 값은 그 
트랜잭션이 변경시킨 블럭에 반영되고, 그 데이타화일의 가장 최근의 SCN은 
데이타화일의 헤더(header)에 기록된다.

(2) statement-level read consistent
하나의 query는 그 query가 시작되어 데이타를 읽기 시작하면, 모든 데이타를 
읽어 query가 끝날 때까지 일관된 상태를 유지한다. 즉 query가 진행되는 동안 
다른 트랜잭션이 읽고자하는 데이타를 변경하더라도 그 query는 변경 이전의 
데이타 값을 읽게 된다.
데이타들이 query가 시작될 때와 같은 시점인지는 SCN을 통해 관리된다. 즉 
SCN이 10인 상태에서 query가 시작되었다면 query가 진행되는 동안 항상 SCN이 
10이하 상태의 데이타만을 읽게 되며, 이것은 롤백 세그먼트(rollback segment)
를 이용하여 read consistent snapshot을 구성함으로써 가능하다.

(3) read consistent snapshot (read consistent view)
트랜잭션이 변경작업을 수행할 때 마다, 오라클은 변경 작업이 이루어지기 
전의 before image(snapshot)을 롤백 세그먼트에 저장해둔다. 한 트랜잭션이 
commit되기 전에 변경된 데이타를 다른 트랜잭션이 읽거나 수정하고자 한다면, 
롤백 세그먼트의 정보를 이용하여 read consistent snapshot을 구성한 후 이 
데이타값을 이용하여 operation을 수행한다. 
또한 (2)에서 설명한 statement-level read consistent를 이루기 위해서도 
query가 진행되는 동안 읽고자 하는 블럭의 SCN이 증가하면, 롤백 세그먼트의 
정보를 이용하여 원하는 SCN상태의 read consistent snapshot을 구성한 후 
데이타를 읽게 된다.

(4) rollback segment의 wrap around/overwrite
롤백 세그먼트는 하나의 롤백 세그먼트를 여러개의 트랜잭션이 함께 사용하며, 
하나의 extent도 여러개의 트랜잭션이 동시에 사용가능하다. 단 각 블럭은 
하나의 트랜잭션에 할당된다. 트랜잭션들이 사용 중인 extent에 정보를 저장
하고 다음 extent가 필요하면, 해당 롤백 세그먼트에 이미 할당되어 있는 다음 
extent가 active한 undo 정보를 가지고 있는지를 검사한다. active한 undo 
정보를 담고 있지 않은 다음 extent가 current extent가 되며, 트랜잭션들은 
이 extent에 undo image를 저장한다. 할당된 맨 마지막 extent를 확인하게 
되면, 다시 첫번째 extent부터 extent로 돌아와 다시 사용하는 것을 wrap 
around라고, 모두 commit된 트랜잭션의 정보만 담고 있는 extent는 overwrite
된다.
이렇게 롤백 세그먼트의 undo image를 담고 있는 블럭 뿐 아니라 롤백 세그먼트 
헤더 내의 트랜잭션 테이블의 엔트리도 wrap around/overwrite될 수 있다. 
트랜잭션 테이블은 고정된 수의 엔트리를 가지고 있으며, 트랜잭션이 이미 
COMMITTED된 엔트리는 비어있는 것으로 인식하여 다음 트랜잭션이 사용 가능하게 
된다.

댓글 없음:

댓글 쓰기