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

3.5 KiB

id, title, category, status, source_trust_level, verification_status, created_at, updated_at, tags, tech_stack, applied_in, aliases
id title category status source_trust_level verification_status created_at updated_at tags tech_stack applied_in aliases
js-structured-clone JS Structured Clone & Deep Equality 함정 Coding draft B conceptual 2026-05-09 2026-05-09
javascript
deep-clone
structured-clone
vibe-coding
language applicable_to
JavaScript / TypeScript
Web
Node
deep clone
JSON.parse hack
structuredClone
lodash isEqual

Structured Clone & Deep Equality

JSON.parse(JSON.stringify(x)) 가 deep clone 의 default 였던 시절은 끝. structuredClone() 가 native (Node 17+/모던 브라우저). 단 함수, DOM, class instance 일부는 여전히 못 함.

📖 핵심 개념

  • structuredClone: HTML structured clone algorithm. 깊은 복사. circular ref OK. Map/Set/Date/RegExp/TypedArray 지원. 함수/Symbol/DOM 못 함.
  • JSON 방식: 함수/undefined/Date 손실, circular 폭사. 절대 production default X.
  • Lodash _.cloneDeep: 거의 모든 형식 지원. 외부 의존성.

💻 코드 패턴

기본 deep clone

const original = { a: 1, b: { c: [1, 2, 3] }, d: new Date() };
const clone = structuredClone(original);
clone.b.c.push(4);
// original.b.c === [1,2,3] 그대로

못 하는 것

const obj = {
  fn: () => 1,                     // ❌ throws DataCloneError
  el: document.body,               // ❌ DOM
  cls: new MyClass(),              // ❌ class instance (plain prototype 만)
  sym: Symbol('s'),                // ❌
};
structuredClone(obj); // 폭사

Class instance 클로닝

class User {
  constructor(public name: string, public meta: object) {}
}

// structuredClone 은 prototype 잃음
const u = new User('a', { x: 1 });
const c = structuredClone(u); // c instanceof User === false

// 해결: clone 메서드 직접
class User {
  clone(): User { return new User(this.name, structuredClone(this.meta)); }
}

Deep equality

import { dequal } from 'dequal'; // 또는 lodash isEqual

dequal({ a: 1, b: [2] }, { a: 1, b: [2] }); // true

// Reference equality (===)는 빠르지만 객체 내부 같은지 모름
// JSON.stringify 비교는 키 순서 / NaN / Date 정확 X

Circular safe stringify

import { stringify } from 'safe-stable-stringify';
// circular 있으면 [Circular] 마커

🤔 의사결정 기준

데이터 도구
Plain object / array / Date / Map / Set structuredClone
함수 / DOM / class instance 직접 clone 메서드 + structuredClone 재귀
비교 (deep eq) dequal 또는 lodash isEqual
빠른 immutability + structural sharing Immer / immutable.js
직렬화 (network, storage) JSON or msgpack — 의도적 손실 OK

안티패턴

  • JSON.parse(JSON.stringify(x)) 를 deep clone default: Date → string, undefined 손실, circular 폭사. 알면서 쓰는 경우만.
  • Object.assign({}, x) 으로 deep clone 기대: shallow.
  • {...x} 로 deep clone 기대: shallow.
  • 가변 객체에 structuredClone 매번: 성능. 진짜 필요할 때만 clone.
  • lodash isEqual 매 렌더: 큰 객체 비교 비용. memo + selector.
  • NaN === NaN false 예상 못함: deep eq 라이브러리 사용.
  • TypedArray byteOffset 바뀜: structuredClone OK 지만 이후 backing buffer 다름.

🤖 LLM 활용 힌트

  • "deep clone = structuredClone, deep eq = dequal" 디폴트.
  • 함수 / class instance 가 있으면 명시적 clone 메서드.

🔗 관련 문서