イベントループ
Node.js のシングルスレッドで非同期 I/O を実現する実行モデル
Node.js非同期
イベントループとは
イベントループは、Node.js のシングルスレッドで非同期 I/O を実現する実行モデルである。コールスタック、タスクキュー、マイクロタスクキューを巡回し、非同期コールバックを順番に実行する。
動作の流れ
1. コールスタックが空か確認
2. マイクロタスクキュー (Promise) を全て実行
3. タスクキュー (setTimeout, I/O) から 1 つ実行
4. 1 に戻る
┌─────────────────────────┐
│ コールスタック │ ← 同期コードを実行
└────────────┬────────────┘
↓
┌─────────────────────────┐
│ マイクロタスクキュー │ ← Promise.then, queueMicrotask
└────────────┬────────────┘
↓
┌─────────────────────────┐
│ タスクキュー │ ← setTimeout, setInterval, I/O
└─────────────────────────┘
実行順序
console.log('1'); // 同期
setTimeout(() => console.log('2'), 0); // タスクキュー
Promise.resolve().then(() => console.log('3')); // マイクロタスク
console.log('4'); // 同期
// 出力: 1, 4, 3, 2
// 同期 → マイクロタスク → タスクキュー の順
Node.js のイベントループフェーズ
| フェーズ | 処理内容 |
|---|---|
| timers | setTimeout, setInterval のコールバック |
| pending callbacks | I/O コールバック |
| poll | 新しい I/O イベントを取得 |
| check | setImmediate のコールバック |
| close callbacks | socket.on('close') 等 |
ブロッキングの問題
// ❌ イベントループをブロック (他のリクエストが処理できない)
function heavyComputation() {
for (let i = 0; i < 1e9; i++) { /* CPU 集約的な処理 */ }
}
// ✅ Worker Threads で別スレッドに逃がす
import { Worker } from 'worker_threads';
const worker = new Worker('./heavy.js');
シングルスレッド vs マルチスレッド
| 観点 | Node.js (イベントループ) | Java (スレッドプール) |
|---|---|---|
| I/O 処理 | 高速 (ノンブロッキング) | スレッドごとにブロック |
| CPU 処理 | 苦手 (シングルスレッド) | 得意 (マルチスレッド) |
| メモリ | 少ない | スレッドごとにスタック |
| 複雑さ | コールバック/Promise | デッドロック、競合 |
詳しくは関連書籍を参照。