프로젝트에서 LOCK 이 자주 걸리는 문제가 발생하여 연구하던 중에 아래의 sql 문을 이용하여 lock 세션을 찾아서 관리할 수 있다는 것을 알았다. 그러나 이 SQL 은 System 유저만이 사용할 수 있다는 제약이 있다.
LOCK 이 발생하게 되면 프로그램이 동작하다가 멈추고 기다리게 된다. 원인을 파악하는데 시간이 허비하게 되는데 알고 보니 테이블에 LOCK이 걸린 것이었다면 허탈함을 금할 수 없다. 자주 LOCK 이 걸리는 테이블이라면 SELECT 할 때 WAIT 타임을 1 정도로 주면 1초동안 LOCK 이 풀리기를 대기하다가 오류 처리되므로 개발자가 쉽게 LOCK 걸린 상황을 인지할 수 있다.
-- 락걸린 테이블 확인
SELECT DO.OBJECT_NAME, DO.OWNER, DO.OBJECT_TYPE, DO.OWNER,
VO.XIDUSN, VO.SESSION_ID, VO.LOCKED_MODE
FROM V$LOCKED_OBJECT VO, DBA_OBJECTS DO
WHERE VO.OBJECT_ID = DO.OBJECT_ID;
-- 해당 테이블에 LOCK 이 걸렸는지.
SELECT A.SID, A.SERIAL#, B.TYPE, C.OBJECT_NAME
FROM V$SESSION A, V$LOCK B, DBA_OBJECTS C
WHERE A.SID = B.SID AND B.ID1 = C.OBJECT_ID
AND B.TYPE='TM' AND C.OBJECT_NAME IN ('TB_CO_GENO');
-- 락발생 사용자와 SQL, OBJECT 조회
SELECT DISTINCT X.SESSION_ID, A.SERIAL#, D.OBJECT_NAME, A.MACHINE, A.TERMINAL,
A.PROGRAM, B.ADDRESS, B.PIECE, B.SQL_TEXT
FROM V$LOCKED_OBJECT X, V$SESSION A, V$SQLTEXT B, DBA_OBJECTS D
WHERE X.SESSION_ID = A.SID AND
X.OBJECT_ID = D.OBJECT_ID AND A.SQL_ADDRESS = B.ADDRESS
ORDER BY B.ADDRESS, B.PIECE;
-- 현재 접속자의 SQL 분석
SELECT DISTINCT A.SID, A.SERIAL#,
A.MACHINE, A.TERMINAL, A.PROGRAM,
B.ADDRESS, B.PIECE, B.SQL_TEXT
FROM V$SESSION A, V$SQLTEXT B
WHERE A.SQL_ADDRESS = B.ADDRESS
ORDER BY A.SID, A.SERIAL#, B.ADDRESS, B.PIECE
-- 락 세션 죽이기
SELECT A.SID, A.SERIAL#
FROM V$SESSION A, V$LOCK B,
DBA_OBJECTS C
WHERE A.SID = B.SID
AND B.ID1 = C.OBJECT_ID
AND B.TYPE = 'TM'
AND C.OBJECT_NAME = 'TB_CO_GENO'
SID SERIAL#
--- -------
5 1
6 1
2. 다음 명령으로 SESSION들을 KILL한다. ALTER SYSTEM KILL SESSION 'SESSION_ID, SERIAL#';
SQL> alter system kill session '5, 1';
SQL> alter system kill session '6, 1';
-- 락 세션 죽이는 sql 문
SELECT DISTINCT X.SESSION_ID, A.SERIAL#, D.OBJECT_NAME, A.MACHINE, A.TERMINAL,
A.PROGRAM, A.LOGON_TIME, 'ALTER SYSTEM KILL SESSION'''||A.SID||', '||A.SERIAL#||''';'
FROM GV$LOCKED_OBJECT X, GV$SESSION A, DBA_OBJECTS D
WHERE X.SESSION_ID = A.SID AND X.OBJECT_ID = D.OBJECT_ID
ORDER BY LOGON_TIME;
-------------------------
1. Parent-Child 관계로 묶인 테이블에서 Child 테이블에 Index가 없는 상황에서 Child 테이블을 수정하게 되면 Parent테이블에 TABLE LEVEL SHARE LOCK이 걸리게 되어서 Parent 테이블에 대한 모든 Update가 금지된다.
2. 블럭의 PCTFREE가 매우 작다면 한 블럭에 여러개의 레코드가 들어 있기 때문에 한 블럭에 과도한 트랜잭션이 들어와서 블럭의 Transaction Layer가 Release 되기를 기다리게 되는 경우도 있다.
3. Transaction을 직접 처리하는 경우 (@Transaction을 사용하지 않고 @Override 를 사용) commit 이나 rollback 하지 않고 바로 빠져 나가는 경우에도 lock 이 발생될 수 있다. 거래가 바로 빠져 나가는 경우는 exception 이 발생했으나 catch 를 제대로 하지 못 한 경우가 흔하다.
다음은 <http://www.medianart.com/288> 에서 퍼온 글입니다. 내가 궁금하던 것을 잘 정리해 놓으셨군요~
LOCK 이 발생하게 되면 프로그램이 동작하다가 멈추고 기다리게 된다. 원인을 파악하는데 시간이 허비하게 되는데 알고 보니 테이블에 LOCK이 걸린 것이었다면 허탈함을 금할 수 없다. 자주 LOCK 이 걸리는 테이블이라면 SELECT 할 때 WAIT 타임을 1 정도로 주면 1초동안 LOCK 이 풀리기를 대기하다가 오류 처리되므로 개발자가 쉽게 LOCK 걸린 상황을 인지할 수 있다.
-- 락걸린 테이블 확인
SELECT DO.OBJECT_NAME, DO.OWNER, DO.OBJECT_TYPE, DO.OWNER,
VO.XIDUSN, VO.SESSION_ID, VO.LOCKED_MODE
FROM V$LOCKED_OBJECT VO, DBA_OBJECTS DO
WHERE VO.OBJECT_ID = DO.OBJECT_ID;
-- 해당 테이블에 LOCK 이 걸렸는지.
SELECT A.SID, A.SERIAL#, B.TYPE, C.OBJECT_NAME
FROM V$SESSION A, V$LOCK B, DBA_OBJECTS C
WHERE A.SID = B.SID AND B.ID1 = C.OBJECT_ID
AND B.TYPE='TM' AND C.OBJECT_NAME IN ('TB_CO_GENO');
-- 락발생 사용자와 SQL, OBJECT 조회
SELECT DISTINCT X.SESSION_ID, A.SERIAL#, D.OBJECT_NAME, A.MACHINE, A.TERMINAL,
A.PROGRAM, B.ADDRESS, B.PIECE, B.SQL_TEXT
FROM V$LOCKED_OBJECT X, V$SESSION A, V$SQLTEXT B, DBA_OBJECTS D
WHERE X.SESSION_ID = A.SID AND
X.OBJECT_ID = D.OBJECT_ID AND A.SQL_ADDRESS = B.ADDRESS
ORDER BY B.ADDRESS, B.PIECE;
-- 현재 접속자의 SQL 분석
SELECT DISTINCT A.SID, A.SERIAL#,
A.MACHINE, A.TERMINAL, A.PROGRAM,
B.ADDRESS, B.PIECE, B.SQL_TEXT
FROM V$SESSION A, V$SQLTEXT B
WHERE A.SQL_ADDRESS = B.ADDRESS
ORDER BY A.SID, A.SERIAL#, B.ADDRESS, B.PIECE
-- 락 세션 죽이기
SELECT A.SID, A.SERIAL#
FROM V$SESSION A, V$LOCK B,
DBA_OBJECTS C
WHERE A.SID = B.SID
AND B.ID1 = C.OBJECT_ID
AND B.TYPE = 'TM'
AND C.OBJECT_NAME = 'TB_CO_GENO'
SID SERIAL#
--- -------
5 1
6 1
2. 다음 명령으로 SESSION들을 KILL한다. ALTER SYSTEM KILL SESSION 'SESSION_ID, SERIAL#';
SQL> alter system kill session '5, 1';
SQL> alter system kill session '6, 1';
-- 락 세션 죽이는 sql 문
SELECT DISTINCT X.SESSION_ID, A.SERIAL#, D.OBJECT_NAME, A.MACHINE, A.TERMINAL,
A.PROGRAM, A.LOGON_TIME, 'ALTER SYSTEM KILL SESSION'''||A.SID||', '||A.SERIAL#||''';'
FROM GV$LOCKED_OBJECT X, GV$SESSION A, DBA_OBJECTS D
WHERE X.SESSION_ID = A.SID AND X.OBJECT_ID = D.OBJECT_ID
ORDER BY LOGON_TIME;
자주 발생하는 LOCK 문제들
-------------------------
1. Parent-Child 관계로 묶인 테이블에서 Child 테이블에 Index가 없는 상황에서 Child 테이블을 수정하게 되면 Parent테이블에 TABLE LEVEL SHARE LOCK이 걸리게 되어서 Parent 테이블에 대한 모든 Update가 금지된다.
2. 블럭의 PCTFREE가 매우 작다면 한 블럭에 여러개의 레코드가 들어 있기 때문에 한 블럭에 과도한 트랜잭션이 들어와서 블럭의 Transaction Layer가 Release 되기를 기다리게 되는 경우도 있다.
3. Transaction을 직접 처리하는 경우 (@Transaction을 사용하지 않고 @Override 를 사용) commit 이나 rollback 하지 않고 바로 빠져 나가는 경우에도 lock 이 발생될 수 있다. 거래가 바로 빠져 나가는 경우는 exception 이 발생했으나 catch 를 제대로 하지 못 한 경우가 흔하다.
다음은 <http://www.medianart.com/288> 에서 퍼온 글입니다. 내가 궁금하던 것을 잘 정리해 놓으셨군요~
LOCK의 유형
데이타의 concurrency를 보장하기 위해 오라클은 lock과 transaction을 사용한다.
Lock은 같은 자원을 access하는 사용자들 사이에 상호간에 해를 끼치는 것을 예방하기
위해서 사용되는 메카니즘이다.
Lock의 종류
- Exclusive : lock이 걸린 자원의 공유를 허용하지 않는다.
- Share : 자원에 대해 수행되는 명령의 유형에 따라 lock된 자원의 공유되는 것을 허용한다.
오라클에서의 Lock의 일반적인 범주
- Data or DML (row locks TX and table locks TM) : 동시에 다중의 사용자에 의해
access되는 테이블 데이타의 보호를 위해 사용된다.
- Dictionary or DDL(TD) : 트랜잭션에서 access되는 테이블과 같은 Object의 정의를
보호하기 위해서 사용한다.
- Internal and Latches (RT, MR) : SGA 영역에서의 내부적인 데이타베이스와 메모리
구조를 보호하기 위해 사용한다.
1. TABLE LOCKS
테이블의 특정한 row를 수정하는 문장은 항상 그러한 row에 대해 exclusive row lock
을 획득하고 테이블 lock을 전유한다.
(1) Row Share Table Locks (RS)
- row를 lock 시키고 lock된 테이블을 UPDATE 할 목적이다.
- 모든 row를 SELECT 하려는 다른 트랜잭션을 허용한다.
- 동일 테이블에서 lock 되지 않은 row를 INSERT, UPDATE, DELETE 하는 다른 트랜잭션을 허용한다.
- 테이블에 대한 ROW SHARE, ROW EXCLUSIVE, SHARE, SHARE ROW EXCLUSIVE lock
을 획득하려는 다른 트랜잭션을 허용한다.
- EXCLUSIVE 모드에서의 테이블 locking으로부터 다른 트랜잭션을 예방한다.
(2) Row Exclusive Table Locks (RX)
- row를 lock 시키고 테이블에서 row를 변화시킨다.
- 모든 row를 SELECT 하려는 다른 트랜잭션을 허용한다.
- 동일 테이블에서 lock 되지 않은 row를 INSERT, UPDATE, DELETE 하는 다른
트랜잭션을 허용한다.
- 테이블에 대한 ROW SHARE, ROW EXCLUSIVE lock을 획득하려는 다른 트랜잭션을
허용한다.
- SHARE, EXCLUSIVE, SHARE ROW EXCLUSIVE 모드에서의 테이블 locking으로부터
다른 트랜잭션을 예방한다.
(3) Share Table Locks (S)
- 다른 트랜잭션을 제공하지 않는 테이블에서 row를 INSERT, UPDATE, DELETE
하는 것이 SHARE lock을 홀딩한다.
- 동일 테이블에서 지정된 row를 QUERY 또는 LOCK 하려는 다른 트랜잭션을 허용한다.
- 테이블에 대해 더 나아가 SHARE lock을 얻으려는 다른 트랜잭션을 허용한다.
- EXCLUSIVE 또는 SHARE ROW EXCLUSIVE 모드로 테이블을 locking 하는 것으로
부터 다른 트랜잭션을 보호한다.
(4) Share Row Exclusive Table Locks (SRX)
- 테이블에 row를 INSERT, UPDATE, DELETE 한다.
- 동일 테이블에서 지정된 row를 QUERY 또는 LOCK 하려는 다른 트랜잭션을 허용한다.
- SHARE, EXCLUSIVE 또는 SHARE ROW EXCLUSIVE 모드로 테이블을 locking 하는 것으로
부터 다른 트랜잭션을 보호한다.
(5) Exclusive Table Locks (S)
- 테이블에 row를 INSERT, UPDATE, DELETE 한다.
- 동일 테이블에서 지정된 row를 QUERY하는 것에 대해서만 다른 트랜잭션을
허용한다.
- 어떠한 lock 문장을 내리는 것으로부터 다른 트랜잭션을 보호한다.
2. DICTIONARY LOCKS
Dictionary lock은 Object에 대한 DDL 명령이 수행되는 동안 Object의 정의를 보호한다.
3. INTERNAL LOCKS 과 LATCHES
Latches 와 Internal lock은 메모리 구조를 보호하기 위한 메카니즘이다.
- Latches 는 SGA 영역에 있는 shared data structure를 보호하기 위한 low-level 연속 메카니즘이다.
- Internallock은 data dictionary entry를 보호하고 database file, tablespace, rollback segment를 lock한다.
Latches 와 Internal locks는 데이타베이스 사용자에 의해 컨트롤 될 수 없다.
LOCK에 대한 모니터
lock contention 과 병목현상은 시스템의 효능을 저하시킨다.
1. 모니터 방법
(1) SQL*DBA 모니터 lock display
(2) V$LOCK view
(3) utllockt.sql 이라는 스크립트 화일의 내용을 확인
(lock이 된 자원을 기다리는 사용자가 있는지를 보여줌)
2. SQL*DBA lock monitor 를 사용한 모니터 방법
Lock 모니터의 Resource ID 1 필드는 DBA_OBJECTS 에서의 object_id 이다.
Lock 모니터의 특정한 Resource ID 1 과 일치하는 테이블명을 찾기 위해서는 다음 Query를 사용한다.
SQL> SELECT owner, object_id, object_name, object_type
FROM dba_objects
WHERE object_id = resource id #;
▶ LOCK monitor의 컬럼들
Username : 연결된 사용자의 명
Session ID : 사용자 연결을 식별하기 위해 오라클에서 제공되는 ID
Serial Number : 세션의 시리얼 번호로 각각의 세션을 유일한 번호로 지정하기 위해 Session ID를 함께 사용한다.
Lock Type : TM은 데이타 조작에 대해 dictionary lock을 표시한다.
Resource ID 1 필드는 lock이 걸린 테이블의 ID이다.
테이블명을 찾아보려면 DBA_OBJECTS 쿼리를 사용한다.
TM lock은 트랜잭션 lock이다.
지시하는 row는 lock이 걸리거나 lock 되기 위해 요청된다.
Resource ID 1 : TM 형태의 lock에 대해 DBA_OBJECTS view로 찾은 테이블 ID를 보여준다. 다른 값은 무시될 수 있다.
Resource ID 2 : 내부적으로 lock의 형태를 식별하기 위해 사용한다.
Mode Held : 자원을 쥐고 있는 lock의 모드를 나타낸다.
Mode Requested : 자원에게 요청된 lock의 모드를 나타낸다.
데이타의 concurrency를 보장하기 위해 오라클은 lock과 transaction을 사용한다.
Lock은 같은 자원을 access하는 사용자들 사이에 상호간에 해를 끼치는 것을 예방하기
위해서 사용되는 메카니즘이다.
Lock의 종류
- Exclusive : lock이 걸린 자원의 공유를 허용하지 않는다.
- Share : 자원에 대해 수행되는 명령의 유형에 따라 lock된 자원의 공유되는 것을 허용한다.
오라클에서의 Lock의 일반적인 범주
- Data or DML (row locks TX and table locks TM) : 동시에 다중의 사용자에 의해
access되는 테이블 데이타의 보호를 위해 사용된다.
- Dictionary or DDL(TD) : 트랜잭션에서 access되는 테이블과 같은 Object의 정의를
보호하기 위해서 사용한다.
- Internal and Latches (RT, MR) : SGA 영역에서의 내부적인 데이타베이스와 메모리
구조를 보호하기 위해 사용한다.
1. TABLE LOCKS
테이블의 특정한 row를 수정하는 문장은 항상 그러한 row에 대해 exclusive row lock
을 획득하고 테이블 lock을 전유한다.
(1) Row Share Table Locks (RS)
- row를 lock 시키고 lock된 테이블을 UPDATE 할 목적이다.
- 모든 row를 SELECT 하려는 다른 트랜잭션을 허용한다.
- 동일 테이블에서 lock 되지 않은 row를 INSERT, UPDATE, DELETE 하는 다른 트랜잭션을 허용한다.
- 테이블에 대한 ROW SHARE, ROW EXCLUSIVE, SHARE, SHARE ROW EXCLUSIVE lock
을 획득하려는 다른 트랜잭션을 허용한다.
- EXCLUSIVE 모드에서의 테이블 locking으로부터 다른 트랜잭션을 예방한다.
(2) Row Exclusive Table Locks (RX)
- row를 lock 시키고 테이블에서 row를 변화시킨다.
- 모든 row를 SELECT 하려는 다른 트랜잭션을 허용한다.
- 동일 테이블에서 lock 되지 않은 row를 INSERT, UPDATE, DELETE 하는 다른
트랜잭션을 허용한다.
- 테이블에 대한 ROW SHARE, ROW EXCLUSIVE lock을 획득하려는 다른 트랜잭션을
허용한다.
- SHARE, EXCLUSIVE, SHARE ROW EXCLUSIVE 모드에서의 테이블 locking으로부터
다른 트랜잭션을 예방한다.
(3) Share Table Locks (S)
- 다른 트랜잭션을 제공하지 않는 테이블에서 row를 INSERT, UPDATE, DELETE
하는 것이 SHARE lock을 홀딩한다.
- 동일 테이블에서 지정된 row를 QUERY 또는 LOCK 하려는 다른 트랜잭션을 허용한다.
- 테이블에 대해 더 나아가 SHARE lock을 얻으려는 다른 트랜잭션을 허용한다.
- EXCLUSIVE 또는 SHARE ROW EXCLUSIVE 모드로 테이블을 locking 하는 것으로
부터 다른 트랜잭션을 보호한다.
(4) Share Row Exclusive Table Locks (SRX)
- 테이블에 row를 INSERT, UPDATE, DELETE 한다.
- 동일 테이블에서 지정된 row를 QUERY 또는 LOCK 하려는 다른 트랜잭션을 허용한다.
- SHARE, EXCLUSIVE 또는 SHARE ROW EXCLUSIVE 모드로 테이블을 locking 하는 것으로
부터 다른 트랜잭션을 보호한다.
(5) Exclusive Table Locks (S)
- 테이블에 row를 INSERT, UPDATE, DELETE 한다.
- 동일 테이블에서 지정된 row를 QUERY하는 것에 대해서만 다른 트랜잭션을
허용한다.
- 어떠한 lock 문장을 내리는 것으로부터 다른 트랜잭션을 보호한다.
2. DICTIONARY LOCKS
Dictionary lock은 Object에 대한 DDL 명령이 수행되는 동안 Object의 정의를 보호한다.
3. INTERNAL LOCKS 과 LATCHES
Latches 와 Internal lock은 메모리 구조를 보호하기 위한 메카니즘이다.
- Latches 는 SGA 영역에 있는 shared data structure를 보호하기 위한 low-level 연속 메카니즘이다.
- Internallock은 data dictionary entry를 보호하고 database file, tablespace, rollback segment를 lock한다.
Latches 와 Internal locks는 데이타베이스 사용자에 의해 컨트롤 될 수 없다.
LOCK에 대한 모니터
lock contention 과 병목현상은 시스템의 효능을 저하시킨다.
1. 모니터 방법
(1) SQL*DBA 모니터 lock display
(2) V$LOCK view
(3) utllockt.sql 이라는 스크립트 화일의 내용을 확인
(lock이 된 자원을 기다리는 사용자가 있는지를 보여줌)
2. SQL*DBA lock monitor 를 사용한 모니터 방법
Lock 모니터의 Resource ID 1 필드는 DBA_OBJECTS 에서의 object_id 이다.
Lock 모니터의 특정한 Resource ID 1 과 일치하는 테이블명을 찾기 위해서는 다음 Query를 사용한다.
SQL> SELECT owner, object_id, object_name, object_type
FROM dba_objects
WHERE object_id = resource id #;
▶ LOCK monitor의 컬럼들
Username : 연결된 사용자의 명
Session ID : 사용자 연결을 식별하기 위해 오라클에서 제공되는 ID
Serial Number : 세션의 시리얼 번호로 각각의 세션을 유일한 번호로 지정하기 위해 Session ID를 함께 사용한다.
Lock Type : TM은 데이타 조작에 대해 dictionary lock을 표시한다.
Resource ID 1 필드는 lock이 걸린 테이블의 ID이다.
테이블명을 찾아보려면 DBA_OBJECTS 쿼리를 사용한다.
TM lock은 트랜잭션 lock이다.
지시하는 row는 lock이 걸리거나 lock 되기 위해 요청된다.
Resource ID 1 : TM 형태의 lock에 대해 DBA_OBJECTS view로 찾은 테이블 ID를 보여준다. 다른 값은 무시될 수 있다.
Resource ID 2 : 내부적으로 lock의 형태를 식별하기 위해 사용한다.
Mode Held : 자원을 쥐고 있는 lock의 모드를 나타낸다.
Mode Requested : 자원에게 요청된 lock의 모드를 나타낸다.
'컴퓨터활용 > 오라클' 카테고리의 다른 글
오라클 테이블의 디스크 용량 예측 (0) | 2011.12.05 |
---|---|
Oracle Locking Survival Guide (0) | 2011.11.30 |
오라클 매뉴얼 (0) | 2011.04.04 |
Golden 에서 접속 오류 날 때 (0) | 2010.06.02 |
INDEX_DESC HINT (0) | 2009.11.07 |