---
id: wiki-2026-0508-total-blocking-time-tbt
title: Total Blocking Time (TBT)
category: 10_Wiki/Topics
status: verified
canonical_id: self
aliases: [TBT, Total Blocking Time]
duplicate_of: none
source_trust_level: A
confidence_score: 0.9
verification_status: applied
tags: [web-vitals, performance, lab-metric, frontend]
raw_sources: []
last_reinforced: 2026-05-10
github_commit: pending
tech_stack:
language: javascript
framework: web-vitals
---
# Total Blocking Time (TBT)
## 매 한 줄
> **"매 FCP 의 TTI 사이 의 long-task blocking 의 sum"**. TBT 의 lab metric 의 user-perceived input responsiveness 의 quantify, 매 each long task (>50ms) 의 50ms-over portion 의 add. 매 2026: INP 의 field metric 의 promote 의 후 의 TBT 의 lab proxy 의 critical, 매 Lighthouse / WebPageTest 의 score 의 driver.
## 매 핵심
### 매 정의
- **Long task**: main thread 의 50ms 의 over 의 continuous 의 block.
- **Blocking portion**: long task duration 의 (duration − 50ms).
- **TBT**: FCP 의 ~ TTI 사이 의 모든 long task 의 blocking portion 의 sum.
- **Threshold (2026)**: Good <200ms · Needs Improvement 200–600ms · Poor >600ms.
### 매 INP 의 차이
- TBT: lab, FCP→TTI window, all long task의 sum.
- INP: field (RUM), 매 user interaction 의 worst (98th %ile-ish), 매 single-event latency.
- 매 correlation 의 high — 매 TBT 의 fix 의 INP 의 usually improve.
### 매 응용
1. CI 의 Lighthouse budget 의 regression gate.
2. Bundle bloat 의 detect (parse/compile time spike).
3. Hydration cost 의 SSR/SSG framework 의 measure.
4. Third-party script 의 main-thread cost 의 audit.
## 💻 패턴
### Measure 의 PerformanceObserver
```javascript
const observer = new PerformanceObserver((list) => {
for (const entry of list.getEntries()) {
const blocking = Math.max(0, entry.duration - 50);
console.log(`Long task: ${entry.duration.toFixed(0)}ms (blocking ${blocking.toFixed(0)}ms)`);
}
});
observer.observe({ type: 'longtask', buffered: true });
```
### Lighthouse CI assertion
```json
{
"ci": {
"assert": {
"assertions": {
"total-blocking-time": ["error", { "maxNumericValue": 200 }],
"interactive": ["warn", { "maxNumericValue": 3500 }]
}
}
}
}
```
### web-vitals lib (TTFB+TBT-ish 의 INP 의 use)
```javascript
import { onINP, onLCP, onCLS, onTTFB } from 'web-vitals';
onINP(({ value, rating }) => analytics.send('inp', { value, rating }));
// 매 TBT 의 field 의 X — 매 lab 의 only
```
### Yield to scheduler (break long task)
```javascript
async function processItems(items) {
for (let i = 0; i < items.length; i++) {
work(items[i]);
if (i % 100 === 0) {
await scheduler.yield(); // 매 Chrome 129+ 의 baseline
}
}
}
```
### Defer non-critical script
```html
```
### Web Worker offload
```javascript
const worker = new Worker('/heavy-parse.js', { type: 'module' });
worker.postMessage(largeJsonString);
worker.onmessage = (e) => updateUI(e.data);
```
### Code split / lazy hydrate
```javascript
// React 19 / Next.js 의 example
import { lazy, Suspense } from 'react';
const Heavy = lazy(() => import('./HeavyChart'));
}>
```
### Long-task budget script
```javascript
let totalBlocking = 0;
const obs = new PerformanceObserver((list) => {
for (const e of list.getEntries()) totalBlocking += Math.max(0, e.duration - 50);
});
obs.observe({ type: 'longtask', buffered: true });
window.addEventListener('load', () => {
if (totalBlocking > 200) console.warn('TBT budget exceeded:', totalBlocking);
});
```
## 매 결정 기준
| 상황 | Approach |
|---|---|
| Bundle parse cost 의 high | code split + dynamic import |
| Hydration block | partial / progressive hydration (Astro, Qwik, React 19 RSC) |
| Third-party script | facade pattern, defer, web worker (Partytown) |
| Heavy compute 의 sync | Web Worker 또는 `scheduler.yield` |
| Re-render cascade | memoization, virtualization |
**기본값**: 매 200ms TBT 의 target — 매 long task 의 50ms 의 budget 의 hold, 매 yield 의 use.
## 🔗 Graph
- 부모: [[Web Vitals]]
## 🤖 LLM 활용
**언제**: 매 CI 의 lab regression 의 detect, 매 PR 의 main-thread cost 의 review.
**언제 X**: 매 real-user experience 의 measure — 매 INP / RUM 의 prefer.
## ❌ 안티패턴
- **TBT 의 only 의 optimize 의 INP 의 ignore**: 매 lab 의 fast, 매 user 의 slow 의 case.
- **Synchronous JSON.parse 의 large payload**: 매 single long task 의 block — 매 stream 또는 worker.
- **`setTimeout(0)` 의 yield 의 substitute**: 매 4ms minimum delay — 매 `scheduler.yield` 의 use.
- **Third-party 의 `