Files
2nd/10_Wiki/Topics/Coding/DB_Transaction_Isolation.md
T
2026-05-09 21:08:02 +09:00

122 lines
4.1 KiB
Markdown

---
id: db-transaction-isolation
title: DB Transaction Isolation Levels — 실전 선택
category: Coding
status: draft
source_trust_level: B
verification_status: conceptual
created_at: 2026-05-09
updated_at: 2026-05-09
tags: [database, transaction, isolation, postgres, vibe-coding]
tech_stack: { language: "SQL / Postgres", applicable_to: ["Backend"] }
applied_in: []
aliases: [SERIALIZABLE, REPEATABLE READ, READ COMMITTED, phantom read]
---
# DB Transaction Isolation
> Postgres default = **READ COMMITTED**. 충분한 경우 많지만 race 가 진짜로 위험한 곳에는 **SERIALIZABLE** + retry. 무지성 SERIALIZABLE 은 throughput 폭락.
## 📖 핵심 개념
4 단계 (SQL standard):
- **READ UNCOMMITTED**: dirty read 허용 (Postgres 는 실제로 READ COMMITTED 처럼 동작).
- **READ COMMITTED**: 디폴트. 같은 트랜잭션 안에서 다른 SELECT 가 다른 결과 가능 (non-repeatable).
- **REPEATABLE READ** (Postgres = snapshot): 트랜잭션 시작 시점 snapshot. 같은 SELECT = 같은 결과.
- **SERIALIZABLE**: 직렬 실행한 것과 같은 결과 보장. 충돌 시 SerializationFailure throw.
## 💻 코드 패턴
### 명시적 isolation
```sql
BEGIN ISOLATION LEVEL SERIALIZABLE;
-- ...
COMMIT;
```
### Node + pg
```ts
await client.query('BEGIN ISOLATION LEVEL SERIALIZABLE');
try {
// ... business logic ...
await client.query('COMMIT');
} catch (e: any) {
await client.query('ROLLBACK');
if (e.code === '40001') { // serialization_failure
// 재시도 가능
}
throw e;
}
```
### Race condition 예시 — 잔액 차감
```sql
-- ❌ READ COMMITTED 에서 race
BEGIN;
SELECT balance FROM accounts WHERE id = 1; -- 100
-- 다른 트랜잭션이 동시 동일 작업 진행
UPDATE accounts SET balance = balance - 50 WHERE id = 1; -- 50
COMMIT;
-- 두 트랜잭션 모두 50 으로 끝나는 게 아니라 0 으로 끝남 (UPDATE 의 expression 은 안전)
-- BUT: 절차상 잔액 < 50 검사를 SELECT 후 했다면 둘 다 통과 → 음수 가능
```
해결 방법 4가지:
```sql
-- 1) 단일 UPDATE 에서 검사 + 차감 (가장 단순)
UPDATE accounts SET balance = balance - 50
WHERE id = 1 AND balance >= 50
RETURNING balance;
-- rowCount = 0 → 잔액 부족
```
```sql
-- 2) SELECT FOR UPDATE — 잠금
BEGIN;
SELECT balance FROM accounts WHERE id = 1 FOR UPDATE; -- 다른 tx 블록
UPDATE accounts SET balance = balance - 50 WHERE id = 1;
COMMIT;
```
```sql
-- 3) SERIALIZABLE — 직렬 보장
BEGIN ISOLATION LEVEL SERIALIZABLE;
SELECT balance FROM accounts WHERE id = 1;
UPDATE accounts SET balance = balance - 50 WHERE id = 1;
COMMIT;
-- 충돌 시 SerializationFailure → app 에서 재시도
```
```sql
-- 4) Optimistic — version
UPDATE accounts SET balance = balance - 50, version = version + 1
WHERE id = 1 AND version = $expected;
```
## 🤔 의사결정 기준
| 상황 | 권장 |
|---|---|
| 일반 CRUD | READ COMMITTED (default) |
| 복잡 read-write 한 트랜잭션 안에서 일관성 필요 | REPEATABLE READ |
| 잔액 / 재고 / 좌석 예약 (strict) | SERIALIZABLE + retry, OR 단일 UPDATE 표현식 |
| Read-only report (스냅샷 일관) | REPEATABLE READ |
| 단순 검사+수정 | `UPDATE ... WHERE 조건` 한 줄로 |
## ❌ 안티패턴
- **모든 곳 SERIALIZABLE**: throughput 폭락. 정말 필요한 곳만.
- **SERIALIZABLE 인데 retry 없음**: SerializationFailure 에 사용자 에러 노출.
- **트랜잭션 안에서 외부 API 호출**: 트랜잭션 길어짐 → lock contention.
- **lock 순서 일관성 없음**: deadlock. 항상 같은 순서로 row 잠금 (보통 ID ASC).
- **트랜잭션 너무 큼 (수천 row 변경)**: 다른 트랜잭션 blocking. 작은 batch.
- **autocommit 끄고 commit 잊음**: 영구 lock + 메모리 leak. 명시적 commit/rollback.
- **READ UNCOMMITTED 가정**: Postgres 에서 동작 다름. 디폴트 READ COMMITTED 가정.
## 🤖 LLM 활용 힌트
- "단일 row 검사+수정은 한 UPDATE 표현식. 여러 row 일관성 필요면 SERIALIZABLE + retry" 명시.
- 트랜잭션 안에서 외부 호출 금지 강제.
## 🔗 관련 문서
- [[Optimistic_Concurrency_Control]]
- [[DB_Connection_Pool]]