--- id: db-partitioning-patterns title: Partitioning — Range / List / Hash category: Coding status: draft source_trust_level: B verification_status: conceptual created_at: 2026-05-09 updated_at: 2026-05-09 tags: [database, partitioning, postgres, vibe-coding] tech_stack: { language: "SQL / Postgres", applicable_to: ["Backend"] } applied_in: [] aliases: [partitioning, range partition, partition pruning, table inheritance] --- # Table Partitioning > 거대 테이블 (10M+ rows / 100GB+) 을 작은 파티션으로 분할. **인덱스 작아짐 + 오래된 파티션 통째 drop = 빠른 만료**. Postgres 11+ declarative partitioning. ## 📖 핵심 개념 - Range: 시간 / 숫자 범위 (events_2026_05). - List: 카테고리 / 지역 (users_kr, users_us). - Hash: 균일 분산 (write 부하 분산). - Pruning: WHERE 조건이 파티션 키와 맞으면 다른 파티션 skip. ## 💻 코드 패턴 ### Range partition (시간) ```sql CREATE TABLE events ( id BIGSERIAL, user_id UUID, event_type TEXT, created_at TIMESTAMPTZ NOT NULL, PRIMARY KEY (id, created_at) ) PARTITION BY RANGE (created_at); CREATE TABLE events_2026_04 PARTITION OF events FOR VALUES FROM ('2026-04-01') TO ('2026-05-01'); CREATE TABLE events_2026_05 PARTITION OF events FOR VALUES FROM ('2026-05-01') TO ('2026-06-01'); -- default (필수: 안 맞는 row 의 안전망) CREATE TABLE events_default PARTITION OF events DEFAULT; ``` ### Pruning 확인 ```sql EXPLAIN SELECT * FROM events WHERE created_at >= '2026-05-01' AND created_at < '2026-05-10'; -- "Append" 안에 events_2026_05 만 보여야. default 도 안 보여야. ``` ### 자동 파티션 생성 (pg_partman) ```sql SELECT partman.create_parent( p_parent_table => 'public.events', p_control => 'created_at', p_type => 'native', p_interval => 'monthly', p_premake => 3 -- 3개월 미리 생성 ); ``` ### 오래된 파티션 drop = 빠른 만료 ```sql -- 1년 지난 events 삭제 = DELETE 가 아니라 DROP DROP TABLE events_2025_05; -- vacuum 부담 X, 즉시. ``` ### List partition ```sql CREATE TABLE orders ( id UUID, region TEXT NOT NULL, ... ) PARTITION BY LIST (region); CREATE TABLE orders_kr PARTITION OF orders FOR VALUES IN ('KR'); CREATE TABLE orders_us PARTITION OF orders FOR VALUES IN ('US'); CREATE TABLE orders_other PARTITION OF orders DEFAULT; ``` ### Hash partition (write 분산) ```sql CREATE TABLE accounts (...) PARTITION BY HASH (id); CREATE TABLE accounts_p0 PARTITION OF accounts FOR VALUES WITH (modulus 4, remainder 0); CREATE TABLE accounts_p1 PARTITION OF accounts FOR VALUES WITH (modulus 4, remainder 1); CREATE TABLE accounts_p2 PARTITION OF accounts FOR VALUES WITH (modulus 4, remainder 2); CREATE TABLE accounts_p3 PARTITION OF accounts FOR VALUES WITH (modulus 4, remainder 3); ``` ### 인덱스 — 자동으로 각 파티션에 ```sql CREATE INDEX events_user ON events (user_id); -- Postgres 11+ : 자동으로 모든 파티션에 적용 ``` ### Constraint exclusion 확인 ```sql SET enable_partition_pruning = on; EXPLAIN ANALYZE SELECT count(*) FROM events WHERE user_id = $1 AND created_at >= '2026-05-01'; ``` ## 🤔 의사결정 기준 | 상황 | 파티션 종류 | |---|---| | 시계열 (이벤트, 로그, 메트릭) | Range (날짜) | | 다국가 / 다지역 | List (region) | | 단일 큰 테이블 균등 write | Hash | | TTL / 자동 만료 | Range + drop old partitions | | 멀티테넌트 큰 차이 | List (tenant_id) | | 천만 미만 | 파티션 X — 인덱스로 충분 | ## ❌ 안티패턴 - **PK 가 파티션 키 안 포함**: 못 함 — PK 에 partition column 포함 (composite PK). - **Default 파티션 누락**: 범위 밖 INSERT 실패. - **모든 파티션 동시 query**: pruning 안 됨 — WHERE 에 partition column 포함. - **Cross-partition unique constraint**: 안 됨. 앱 레벨에서. - **파티션 너무 많음 (1000+)**: planner 오버헤드. 적당히 (<100). - **Partition pruning 환경 검사 안 함**: 잘못 만들면 아무 효과 X. - **수동 파티션 생성 잊음**: pg_partman 또는 cron. ## 🤖 LLM 활용 힌트 - Postgres 11+ declarative partitioning 우선. - Range + 자동 생성 (pg_partman) + drop 으로 만료. - WHERE 에 partition column 항상. ## 🔗 관련 문서 - [[DB_Sharding_Strategies]] - [[DB_Soft_Delete_Patterns]] - [[Postgres_Performance_Tuning]]