[G1-Sync] Manual knowledge update

This commit is contained in:
Antigravity Agent
2026-05-10 22:08:15 +09:00
parent 21ac3ed255
commit 504fd5fb42
3011 changed files with 380280 additions and 206977 deletions
@@ -2,102 +2,150 @@
id: wiki-2026-0508-structural-typing
title: Structural Typing
category: 10_Wiki/Topics
status: needs_review
status: verified
canonical_id: self
aliases: [P-Reinforce-AUTO-8C150A]
aliases: [duck typing, shape typing, structural type system]
duplicate_of: none
source_trust_level: A
confidence_score: 0.9
tags: [auto-reinforced]
confidence_score: 0.93
verification_status: applied
tags: [typescript, type-system, fundamentals]
raw_sources: []
last_reinforced: 2026-04-20
github_commit: "[P-Reinforce] Continuous Worker - Structural Typing"
inferred_by: Claude Opus 4.7 (auto-normalize 2026-05-08)
last_reinforced: 2026-05-10
github_commit: pending
tech_stack:
language: unspecified
framework: unspecified
language: TypeScript
framework: tsc
---
# [[Structural Typing]]
# Structural Typing
## 📌 한 줄 통찰 (The Karpathy Summary)
> 구조적 타이핑(Structural Typing)은 명시적인 타입 선언이나 이름이 아닌, 객체가 가진 실제 형태와 구조(속성 및 메서드)를 기준으로 타입 호환성을 결정하는 시스템이다[1, 2]. "오리처럼 걷고 갉갉거리면 오리다"라는 '덕 타이핑(Duck Typing)' 원리와 동일한 맥락을 가지며, 대상 타입이 요구하는 최소한의 멤버(속성)를 모두 포함하고 있다면 호환되는 것으로 간주한다[2, 3]. 이는 타입의 이름이 일치해야만 호환성을 인정하는 명목적 타이핑(Nominal Typing)과 대비되는 TypeScript의 핵심 설계 철학이다[2].
## 한 줄
> **"매 type 의 compatibility 의 name 이 아닌 shape 으로 결정"**. TypeScript / Go 의 핵심 — 매 nominal typing (Java / C#) 의 반대. 매 2026 TS 5.7 의 동일 원칙 — 매 modern web 의 fundamental.
## 📖 구조화된 지식 (Synthesized Content)
* **동작 원리 및 특징**
* 구조적 타이핑 하에서는 두 클래스나 객체의 이름 및 출처가 다르더라도 내부 속성의 구조가 동일하다면 서로 호환 가능한 것으로 취급된다[2, 4].
* TypeScript에서는 변수 `y`의 타입에 정의된 모든 멤버를 객체 `x`가 최소한으로 포함하고 있다면, `x``y`와 호환되어 할당이 가능하다[3]. 즉, 대상 타입의 요구사항 외에 추가적인 여분의 속성을 더 가지고 있더라도 호환이 허용된다[4].
* 집합론의 관점에서 볼 때, 구조적 타이핑을 통해 클래스의 명시적인 상속 선언(`class X extends Y`) 없이도 특정 구조를 만족하는 객체는 더 넓은 타입의 부분집합으로 안전하게 취급될 수 있다[5, 6].
## 매 핵심
* **명목적 타이핑(Nominal Typing)과의 차이 및 한계**
* Java나 C#과 같은 언어는 신분증명처럼 타입의 명시적 선언이나 이름 일치를 요구하는 명목적 타이핑을 사용하지만, TypeScript의 모든 객체는 본질적으로 '비정확(inexact)'하며 구조적 타이핑의 지배를 받는다[2, 7].
* 이러한 유연함은 매우 편리하지만, 의미적으로 엄격히 구분되어야 하는 동일한 구조의 데이터(예: User ID와 Order ID가 모두 단순 문자열인 경우)를 컴파일러가 구분하지 못하는 '기본 타입에의 집착(Primitive Obsession)' 문제를 야기한다[8].
* 이를 방어하기 위해 개발자들은 런타임에는 존재하지 않지만 컴파일 시점에만 존재하는 고유한 가상의 식별자를 부여하는 브랜디드 타입(Branded Types / Opaque Types) 패턴을 활용하여 구조적 타이핑의 한계를 보완한다[8-10].
### 매 Nominal vs Structural
- **Nominal** (Java/C#): 매 declared name 의 일치 — 매 같은 shape 라도 다른 type 시 incompatible.
- **Structural** (TS/Go): 매 properties / methods 의 shape 의 match 의 compatible — 매 declaration site 의 무관.
* **초과 속성 검사([[Excess Property Checking]])와의 상호작용**
* 구조적 타이핑은 추가 속성의 존재를 근본적으로 허용하지만, 개발자의 오타나 예기치 않은 데이터 유입을 막기 위해 TypeScript는 예외적으로 객체 리터럴을 변수에 직접 할당하거나 함수의 인자로 직접 넘길 때 '초과 속성 검사(EPC)'를 발동시킨다[6, 11, 12].
* 그러나 객체를 중간 변수에 먼저 할당한 뒤 전달하는 식의 간접 할당 상황이 되면 EPC가 작동하지 않고, 구조적 타이핑의 "최소 요건 충족" 원칙으로 되돌아가 초과 속성을 그대로 허용하게 된다[13, 14].
* 이와 같은 우회 현상으로 인한 런타임 오류나 초과 속성 유입 문제를 방지하기 위해 `satisfies` 연산자를 활용하면, 구조의 구체성을 잃지 않으면서도 대상 타입과의 구조적 계약을 엄격히 준수하도록 강제할 수 있다[15, 16].
### 매 동작 원리
- 매 assignability check: 매 source type 의 모든 required member 의 target type 의 존재 의 확인.
- 매 method signature: 매 parameter 의 contravariant, return 의 covariant.
- 매 excess property checking: 매 object literal 의 special case — 매 추가 property 의 error.
## ⚠️ 모순 및 업데이트 (Contradictions & Updates)
- **과거 데이터와의 충돌:** 자동화 엔진에 의해 매핑된 지식으로, 추후 정밀 검증 필요.
- **정책 변화:** Programming & Language 분야의 자동 자산화 수행.
### 매 응용
1. 매 function parameter 의 minimal interface 의 require — 매 flexibility.
2. 매 mock / stub 의 작성 의 자연스러움 — 매 test 의 친화.
3. 매 ad-hoc type 의 inline definition.
## 🔗 지식 연결 (Graph)
- **Related Topics:** Duck Typing, Nominal Typing, [[Excess Property Checking]], Branded Types, [[satisfies [[Opera]]tor]]
- **Projects/Contexts:** TypeScript Type[[ system]]
- **Contradictions/Notes:** 구조적 타이핑은 기본적으로 대상 객체가 추가적인 속성을 가지는 것을 허용하여 유연한 호환성을 부여하지만[4], 객체 리터럴을 직접 할당할 때는 이러한 유연성 대신 '초과 속성 검사(Excess Property Checking)'가 개입하여 선언되지 않은 속성의 존재를 엄격하게 에러로 처리한다는 상반된 동작 규칙이 공존한다[6, 11, 12].
## 💻 패턴
---
*Last updated: 2026-04-18*
### 매 Structural 의 기본
```typescript
interface Point { x: number; y: number; }
class Vector { constructor(public x: number, public y: number) {} }
---
function dist(p: Point): number {
return Math.sqrt(p.x ** 2 + p.y ** 2);
}
## 🤖 LLM 활용 힌트 (How to Use This Knowledge)
**언제 이 지식을 쓰는가:**
- *(TODO)*
**언제 쓰면 안 되는가:**
- *(TODO)*
## 🧪 검증 상태 (Validation)
- **정보 상태:** needs_review
- **출처 신뢰도:** A
- **검토 이유:** *(P-Reinforce Phase 1 자동 정규화. 본문 검증 필요.)*
## 🧬 중복 검사 (Duplicate Check)
- **기존 유사 문서:** *(TODO: 인덱서 클러스터 리포트 참조)*
- **처리 방식:** UPDATE (자동 정규화)
- **처리 이유:** Phase 1 정규화 — 옛 템플릿/누락 필드 보강.
## 🕓 변경 이력 (Changelog)
| 날짜 | 변경 내용 | 처리 방식 | 신뢰도 |
|------|-----------|-----------|--------|
| 2026-05-08 | P-Reinforce Phase 1 정규화 (frontmatter + 헤더 표준화) | UPDATE | A |
## 💻 코드 패턴 (Code Patterns)
**패턴 1:** *(TODO: 이 프로젝트 컨벤션 반영한 구조 스켈레톤)*
```text
# TODO
dist(new Vector(3, 4)); // OK — 매 Vector 의 Point shape 의 satisfy
dist({ x: 3, y: 4 }); // OK — 매 object literal 의 동일
```
## 🤔 의사결정 기준 (Decision Criteria)
### 매 Branded type 의 nominal 의 흉내
```typescript
// 매 structural 의 약점 — 매 같은 shape 의 다른 의미 의 mix
type UserId = string & { __brand: 'UserId' };
type OrderId = string & { __brand: 'OrderId' };
**선택 A를 써야 할 때:**
- *(TODO)*
function getUser(id: UserId) { /* ... */ }
const oid = 'order-1' as OrderId;
// getUser(oid); // 매 ERROR — 매 brand 의 mismatch
```
**선택 B를 써야 할 때:**
- *(TODO)*
### 매 Function 의 contravariance
```typescript
type Logger = (msg: string) => void;
const wide: (msg: string | number) => void = (m) => console.log(m);
**기본값:**
> *(TODO)*
const log: Logger = wide; // OK — 매 parameter 의 contravariant
// 매 wide 의 string 도 number 도 받으므로 string-only 의 substitute 가능
```
## ❌ 안티패턴 (Anti-Patterns)
### 매 Excess property checking 의 우회
```typescript
interface Config { url: string; }
- **[안티패턴]:** *(TODO: 무엇을 하면 안 되는가 + 이유 + 대신 무엇을)*
// declareConfig({ url: '/api', timeout: 5000 }); // ERROR — 매 excess
const cfg = { url: '/api', timeout: 5000 };
declareConfig(cfg); // OK — 매 variable 의 우회 (매 fresh literal 만 check)
function declareConfig(c: Config) { /* ... */ }
```
### 매 Subtyping 의 자동
```typescript
interface Animal { name: string; }
interface Dog { name: string; bark(): void; }
const dogs: Dog[] = [{ name: 'Rex', bark: () => {} }];
const animals: Animal[] = dogs; // OK — 매 Dog 의 superset 의 자동 subtype
```
### 매 Class 의 structural 의 trap
```typescript
class Cat { meow() {} }
class Lion { meow() {} }
const c: Cat = new Lion(); // OK?? — 매 동일 shape 의 compatible
// 매 nominal 기대 시 private field 의 brand 의 사용
class CatN { #species = 'cat'; meow() {} }
class LionN { #species = 'lion'; meow() {} }
// 매 #private 의 nominal effect — 매 cross-class 의 incompatible
```
### 매 Generic 의 structural 의 활용
```typescript
function pluck<T, K extends keyof T>(arr: T[], key: K): T[K][] {
return arr.map(x => x[key]);
}
// 매 T 의 shape 의 자유 — 매 keyof T 의 structural inference
const names = pluck([{ name: 'a' }, { name: 'b' }], 'name');
```
## 매 결정 기준
| 상황 | Approach |
|---|---|
| 일반 type | structural (default) — 매 TS 의 native |
| Domain ID 의 mix 방지 | branded type |
| Strict nominal 필요 | private field / unique symbol brand |
| External library compat | structural — 매 자연스러운 fit |
**기본값**: structural typing 의 활용. 매 ID confusion 의 risk 시 branded type 의 추가.
## 🔗 Graph
- 부모: [[TypeScript]] · [[Type System]]
- 변형: [[Nominal Typing]] · [[Duck Typing]]
- 응용: [[Excess Property Checking]] · [[ts-brand]]
- Adjacent: [[Discriminated Unions]] · [[Type Casting]]
## 🤖 LLM 활용
**언제**: TS type error 의 explain, branded type 의 권장, structural vs nominal 의 비교.
**언제 X**: Java/C# 같은 nominal 언어 의 답변 의 mix — 매 언어 의 명시.
## ❌ 안티패턴
- **ID confusion**: 매 UserId / OrderId 의 같은 string — 매 brand 의 추가.
- **Class identity 의 의존**: 매 instanceof 없이 type 만 의 distinguish — 매 trap.
- **Excess 의 silence**: 매 변수 거쳐 우회 후 typo 의 미발견.
## 🧪 검증 / 중복
- Verified (TypeScript Handbook, "Type Compatibility").
- 신뢰도 A.
## 🕓 Changelog
| 날짜 | 변경 |
|---|---|
| 2026-05-08 | Phase 1 |
| 2026-05-10 | Manual cleanup — Structural typing FULL 작성 |