說簡單一點,中間件就是在你的請求和業務邏輯之間做一層攔截。
在 Node.js 中,中間件(Middleware) 是一種函數,它在 請求(Request)到達路由處理器之前,或在 響應(Response)發出之前,對請求進行處理。
🧠 一句話理解:
中間件就是“請求-響應”過程中的 攔截器、處理器或過濾器,可以對請求做日志、驗證、解析、權限、響應包裝等操作。
? 主要特征
在常見的 Node.js Web 框架(如 Express、Koa)中,中間件具有以下特點:
-
本質是一個函數
-
接收
(req, res, next)
這三個參數(Express) -
可以決定是否繼續交給下一個中間件(調用
next()
)
📦 Express 中間件示例
示例:記錄請求日志的中間件
const express = require('express');
const app = express();// 中間件函數
function logger(req, res, next) {console.log(`[${new Date().toISOString()}] ${req.method} ${req.url}`);next(); // 繼續交給下一個中間件
}app.use(logger); // 應用中間件app.get('/', (req, res) => {res.send('Hello, Middleware!');
});app.listen(3000, () => console.log('Server started'));
🚀 中間件的分類
類型 | 說明 |
---|---|
應用級中間件 | 使用 app.use() 或 app.METHOD() 注冊 |
路由級中間件 | 只在特定路由生效 |
錯誤處理中間件 | 帶有 4 個參數 (err, req, res, next) |
內置中間件 | 如 express.json() 、express.static() |
第三方中間件 | 如 body-parser 、cors 、helmet 等 |
🧩 示例:多個中間件串聯
app.use((req, res, next) => {console.log('中間件 A');next();
});app.use((req, res, next) => {console.log('中間件 B');next();
});app.get('/', (req, res) => {console.log('路由處理器');res.send('OK');
});
訪問 /
頁面,輸出:
中間件 A
中間件 B
路由處理器
🔄 Koa 中的中間件機制(異步、洋蔥模型)
Koa 中間件是 async 函數,支持“前進+回退”的洋蔥模型執行機制:
const Koa = require('koa');
const app = new Koa();app.use(async (ctx, next) => {console.log('1. Start');await next(); // 進入下一層console.log('1. End');
});app.use(async (ctx, next) => {console.log('2. Start');await next();console.log('2. End');
});app.use(async ctx => {console.log('3. Handler');ctx.body = 'Hello from Koa';
});app.listen(3000);
輸出順序是:
1. Start
2. Start
3. Handler
2. End
1. End
📌 中間件的用途舉例
目的 | 示例 |
---|---|
請求日志記錄 | morgan , 自定義 logger 中間件 |
權限認證 | 判斷用戶 token 是否有效 |
解析請求體 | body-parser , express.json() |
設置響應頭 | 跨域 CORS 控制 |
捕獲異常 | 錯誤處理中間件 |
靜態資源托管 | express.static('public') |
🧠 總結一句話:
中間件是構成 Node.js Web 框架(如 Express/Koa)核心機制之一,它將請求處理拆分為多個函數鏈式執行,使得邏輯清晰、易于復用和維護。
如果你想自己寫一個中間件系統(類 Express),我可以提供底層實現原理代碼。是否需要?