f8b21af4be
10_Wiki/Topics 대규모 정리: - 오류 캡처/미완성 stub 문서 227개 제거 - 교차폴더 중복 43클러스터 병합 (63파일 → redirect) - 링크명 정규화: 깨진 링크 수정·redirect 직결·개념 매핑 ~2,400건 - 카테고리 MOC 6개 신규 생성 - Graph 섹션 미해결 related-keyword 링크 10,058건 제거 Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
7.2 KiB
7.2 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-웹-워커-이벤트-포워딩-통신-지연-최소화-방법 | 웹 워커 이벤트 포워딩 통신 지연 최소화 방법 | 10_Wiki/Topics | verified | self |
|
none | A | 0.88 | applied |
|
2026-05-10 | pending |
|
웹 워커 이벤트 포워딩 통신 지연 최소화 방법
매 한 줄
"매 main thread ↔ Worker 통신 의 latency 의 4가지 의 source: serialization, structured clone, queue, dispatch — 매 Transferable / SharedArrayBuffer / batching / Comlink RPC 의 mitigation". 매 postMessage 의 default behavior 의 매 bytes 의 deep clone — 매 large payload 의 매 ms 의 단위. 매 2026 의 modern approach 의 OffscreenCanvas, transferable streams, Atomics.
매 핵심
매 Latency Source
- Structured clone — 매 object 의 deep copy, 매 size 의 비례.
- Serialization — 매 V8 의 internal format 의 conversion.
- Event queue — 매 receiver 의 task queue 의 enqueue.
- Dispatch overhead — 매 microtask boundary, 매 ~0.1-0.5ms.
매 Mitigation 전략
- Transferable Objects —
ArrayBuffer,MessagePort,OffscreenCanvas,ReadableStream의 zero-copy 의 ownership transfer. - SharedArrayBuffer + Atomics — 매 shared memory, 매 lock-free 통신.
- Batching — 매 multiple event 의 single message 의 합침.
- MessageChannel — 매 dedicated channel, 매 main thread 의 bypass.
- Comlink — 매 RPC abstraction, 매 ergonomics.
매 Transferable Objects (2026)
ArrayBuffer,MessagePort,ImageBitmap,OffscreenCanvas,RTCDataChannel.ReadableStream,WritableStream,TransformStream(Streams API).VideoFrame,AudioData(WebCodecs).- 매 transfer 후 매 sender 의 access 의 X (neutered).
💻 패턴
Pattern 1: Transferable ArrayBuffer (zero-copy)
// main.ts
const buffer = new ArrayBuffer(10_000_000); // 10MB
const view = new Float32Array(buffer);
view.fill(1.0);
worker.postMessage({ buffer }, [buffer]); // 매 두번째 인자 의 transfer list
console.log(buffer.byteLength); // 0 — 매 neutered
// worker.ts
self.onmessage = (e) => {
const buffer: ArrayBuffer = e.data.buffer;
// 매 zero-copy 의 ownership 의 받음
const view = new Float32Array(buffer);
// ... process
};
Pattern 2: Comlink RPC (ergonomic)
// worker.ts
import * as Comlink from 'comlink';
const api = {
async heavyCompute(data: Float32Array): Promise<Float32Array> {
// ... compute
return result;
},
async streamResults(onChunk: (c: number[]) => void) {
for (let i = 0; i < 100; i++) {
onChunk([Math.random()]);
await new Promise(r => setTimeout(r, 10));
}
},
};
Comlink.expose(api);
// main.ts
import * as Comlink from 'comlink';
const worker = new Worker(new URL('./worker.ts', import.meta.url), { type: 'module' });
const api = Comlink.wrap<typeof api>(worker);
const result = await api.heavyCompute(input);
Pattern 3: SharedArrayBuffer + Atomics (lock-free)
// 매 COOP/COEP header 의 필요 — Cross-Origin-Opener-Policy: same-origin
const sab = new SharedArrayBuffer(1024);
const view = new Int32Array(sab);
worker.postMessage({ sab });
// worker 의 spinlock 의 wait
Atomics.wait(view, 0, 0); // 매 view[0]==0 의 동안 sleep
const data = view[1];
// main 의 signal
view[1] = 42;
view[0] = 1;
Atomics.notify(view, 0, 1); // 매 worker 의 wake
Pattern 4: Batching (event coalescing)
// main.ts
const queue: Event[] = [];
let scheduled = false;
function forward(event: Event) {
queue.push(event);
if (!scheduled) {
scheduled = true;
queueMicrotask(() => {
worker.postMessage({ events: queue.splice(0) });
scheduled = false;
});
}
}
window.addEventListener('mousemove', forward); // 매 60fps 의 single batch
Pattern 5: OffscreenCanvas (rendering off main thread)
// main.ts
const canvas = document.querySelector('canvas')!;
const offscreen = canvas.transferControlToOffscreen();
worker.postMessage({ canvas: offscreen }, [offscreen]);
// worker.ts
self.onmessage = (e) => {
const canvas = e.data.canvas as OffscreenCanvas;
const ctx = canvas.getContext('2d')!;
// 매 main thread 의 block X 의 render
function frame() {
ctx.clearRect(0, 0, canvas.width, canvas.height);
// ... draw
requestAnimationFrame(frame);
}
frame();
};
Pattern 6: MessageChannel (direct port)
const channel = new MessageChannel();
worker1.postMessage({ port: channel.port1 }, [channel.port1]);
worker2.postMessage({ port: channel.port2 }, [channel.port2]);
// 매 worker 끼리 매 main thread 의 bypass 의 직접 통신
Pattern 7: Transferable Stream (2026)
// main.ts
const stream = new ReadableStream({
start(ctrl) {
fetch('/large').then(async r => {
const reader = r.body!.getReader();
while (true) {
const { value, done } = await reader.read();
if (done) break;
ctrl.enqueue(value);
}
ctrl.close();
});
}
});
worker.postMessage({ stream }, [stream]); // 매 zero-copy
// worker.ts
self.onmessage = async (e) => {
for await (const chunk of e.data.stream) {
process(chunk);
}
};
매 결정 기준
| 상황 | 기법 |
|---|---|
| Large binary payload | Transferable ArrayBuffer |
| RPC ergonomics | Comlink |
| Lock-free shared state | SharedArrayBuffer + Atomics |
| High-frequency events | Batching with queueMicrotask |
| Canvas rendering | OffscreenCanvas |
| Worker-to-worker | MessageChannel |
| Streaming data | Transferable streams |
기본값: Transferable ArrayBuffer + Comlink RPC + 매 60fps event 의 batching.
🔗 Graph
- 부모: Web Worker (웹 워커) · 웹 프론트엔드 성능 최적화
- 변형: Service Worker · SharedWorker · Worklet
- Adjacent: Event Loop · Structured Clone · Comlink
🤖 LLM 활용
언제: postMessage 의 large payload 의 transferable 의 변환 권고, COOP/COEP header 의 troubleshoot, batching 의 적절한 frequency 의 제안. 언제 X: 매 native thread (C++ worker) 의 통신 — 매 다른 domain.
❌ 안티패턴
- Large object 의 매 postMessage: 매 deep clone, 매 GC pressure, 매 main thread 의 stall.
- Per-event 매 postMessage (mousemove): 매 queue 의 폭주, 매 jank.
- Atomics 의 SharedArrayBuffer 없음: 매 cross-origin isolation 의 필요.
- Worker 의 dynamic import 의 누락: 매 type: 'module' 의 누락.
- Transferable 의 후 의 access: 매 neutered, 매 byteLength=0.
🧪 검증 / 중복
- Verified (HTML spec: Workers, MDN Transferable Objects, Comlink GitHub 2026).
- 신뢰도 A.
🕓 Changelog
| 날짜 | 변경 |
|---|---|
| 2026-05-08 | Phase 1 |
| 2026-05-10 | Manual cleanup — Transferables, Comlink RPC, SAB+Atomics, batching patterns |