d8a80f6272
이름만 다른(표기 변형) [[위키링크]]를 대상 문서의 canonical 제목으로 치환해 끊겼던 1,200개 링크를 연결. 제목/파일명 정규화 일치만 적용하고 별칭 매칭은 과병합 위험으로 제외(애매성 가드). 원본은 _link_reconcile_backup/ 에 백업. 도구: Datacollect/scripts/link_reconcile_apply.mjs Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
6.1 KiB
6.1 KiB
id, title, category, status, canonical_id, aliases, duplicate_of, source_trust_level, confidence_score, verification_status, tags, raw_sources, last_reinforced, github_commit, tech_stack
| id | title | category | status | canonical_id | aliases | duplicate_of | source_trust_level | confidence_score | verification_status | tags | raw_sources | last_reinforced | github_commit | tech_stack | ||||||||||||
|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
| wiki-2026-0508-open-closed-principle-ocp | Open-Closed Principle (OCP) | 10_Wiki/Topics | verified | self |
|
none | A | 0.9 | applied |
|
2026-05-10 | pending |
|
Open-Closed Principle (OCP)
매 한 줄
"매 extension 에 open, modification 에 closed". Bertrand Meyer (1988) 매 originally inheritance, Robert Martin (1996) 매 abstraction-based redefinition. 새 behavior 매 추가 의 way 로, 매 existing code 의 수정 없이 — strategy / plugin / open enum 등 매 광범위 응용.
매 핵심
매 두 가지 해석
- Meyer's OCP: 매 inheritance — class 매 closed for modification (매 published interface 안정), open for extension (매 subclass).
- Polymorphic OCP (Martin): 매 abstraction (interface) 의존, 매 implementation 추가 만 으로 extend.
매 왜 중요
- 매 regression 위험 줄임 (existing 안 건드림).
- 매 plugin / extension model 가능.
- 매 testability — 매 mock implementation 추가 만.
매 응용
- Strategy pattern — payment processor (Stripe, Toss, PayPal).
- Discriminated union 의 add new variant.
- Plugin architecture (Vite plugin, ESLint rule).
- Visitor pattern.
- Middleware chain (Koa, Express).
💻 패턴
Strategy interface (TS)
interface PaymentProcessor {
charge(amount: number, token: string): Promise<{ id: string }>;
}
class StripeProcessor implements PaymentProcessor {
async charge(amount: number, token: string) {
return { id: `stripe_${Date.now()}` };
}
}
class TossProcessor implements PaymentProcessor {
async charge(amount: number, token: string) {
return { id: `toss_${Date.now()}` };
}
}
// New processor → just add new class. CheckoutService unchanged.
class CheckoutService {
constructor(private processor: PaymentProcessor) {}
pay(amt: number, t: string) { return this.processor.charge(amt, t); }
}
Discriminated union + exhaustiveness
type Shape =
| { kind: "circle"; r: number }
| { kind: "square"; side: number }
| { kind: "triangle"; base: number; height: number };
function area(s: Shape): number {
switch (s.kind) {
case "circle": return Math.PI * s.r ** 2;
case "square": return s.side ** 2;
case "triangle": return (s.base * s.height) / 2;
default: {
const _exhaust: never = s;
throw new Error(_exhaust);
}
}
}
// Add 'pentagon' → TS error in `area`, no silent miss.
Plugin / hook registry
type Hook<T> = (ctx: T) => void | Promise<void>;
class HookRegistry<T> {
private hooks: Hook<T>[] = [];
add(h: Hook<T>) { this.hooks.push(h); }
async run(ctx: T) { for (const h of this.hooks) await h(ctx); }
}
const onUserCreated = new HookRegistry<{ userId: string }>();
onUserCreated.add(({ userId }) => sendWelcomeEmail(userId));
onUserCreated.add(({ userId }) => analytics.track("user_created", userId));
// New behavior → add new hook, no service change.
Visitor pattern
interface NodeVisitor<R> {
visitNumber(n: NumberNode): R;
visitBinary(n: BinaryNode): R;
}
abstract class AstNode { abstract accept<R>(v: NodeVisitor<R>): R; }
class NumberNode extends AstNode {
constructor(public val: number) { super(); }
accept<R>(v: NodeVisitor<R>) { return v.visitNumber(this); }
}
class BinaryNode extends AstNode {
constructor(public op: "+"|"-", public l: AstNode, public r: AstNode) { super(); }
accept<R>(v: NodeVisitor<R>) { return v.visitBinary(this); }
}
// New traversal (printer, evaluator, optimizer) → new visitor class.
Middleware chain
type Middleware<C> = (ctx: C, next: () => Promise<void>) => Promise<void>;
function compose<C>(mws: Middleware<C>[]): (ctx: C) => Promise<void> {
return async (ctx) => {
let i = -1;
const dispatch = async (idx: number): Promise<void> => {
if (idx <= i) throw new Error("next() called multiple times");
i = idx;
const fn = mws[idx];
if (!fn) return;
await fn(ctx, () => dispatch(idx + 1));
};
await dispatch(0);
};
}
const app = compose<{ req: Request; res?: Response }>([
async (c, next) => { console.log("log"); await next(); },
async (c, next) => { c.res = new Response("ok"); await next(); },
]);
매 결정 기준
| 상황 | Approach |
|---|---|
| Multiple swappable algos | Strategy / interface |
| Closed set of variants | Discriminated union + exhaustive switch |
| Open set of behaviors | Plugin / hook registry |
| Tree traversal, multiple ops | Visitor |
| Pipeline | Middleware chain |
| Just one impl, no plans | Don't pre-abstract (YAGNI) |
기본값: 매 second variation 발견 시 abstract — 매 first time 의 over-abstraction X.
🔗 Graph
- 부모: SOLID Principles · Design-Principles
- 변형: Visitor-Pattern
- 응용: Middleware · API Response & State Modeling · Dependency_Injection_(DI)
- Adjacent: DIP · Composition-over-Inheritance
🤖 LLM 활용
언제: 매 add-only domain (payment processor, plugin, parser AST), variant 매 자주 추가. 언제 X: 매 small / unchanging code — 매 abstraction 의 cost > benefit.
❌ 안티패턴
- Premature abstraction: 매 single impl 의 interface — 매 indirection 만 추가, value 0.
- Speculative generality: 매 "혹시 나중에" — YAGNI.
- Inheritance for code reuse: 매 LSP 위반 risk. Composition 우선.
- Open everything: 매 모든 method virtual / hook — 매 cognitive load 폭증.
- Ignore exhaustiveness: 매 switch default fallback — 매 new variant 의 silent skip.
🧪 검증 / 중복
- Verified (Meyer "Object-Oriented Software Construction", Martin "Clean Architecture", refactoring.guru).
- 신뢰도 A.
🕓 Changelog
| 날짜 | 변경 |
|---|---|
| 2026-05-08 | Phase 1 |
| 2026-05-10 | Manual cleanup — strategy, discriminated union, plugin/visitor/middleware 패턴 |