答案:通过洋葱模型的中间件机制实现轻量级http请求库,核心是将请求流程抽象为可插拔函数链。每个中间件接收配置和下一环节函数,支持在调用前后处理逻辑,如日志、认证等。使用reduceRight从右向左组合中间件,形成执行链,最内层调用实际请求方法。提供简洁API如request、get、post,并通过use注册中间件。需注意错误冒泡与配置冻结问题,采用深拷贝或不可变处理确保安全性。
实现一个支持中间件机制的轻量级 HTTP 请求库,核心在于将请求和响应的处理流程抽象成可插拔的函数链。通过中间件机制,开发者可以在不修改核心逻辑的前提下,灵活添加日志、重试、认证、超时等功能。
设计中间件接口
中间件本质上是一个函数,接收请求配置和下一个中间件作为参数,返回一个 promise 形式的响应。结构类似于 Koa 的洋葱模型:
type Middleware = (config: RequestConfig, next: () => Promise
每个中间件可以选择在 next 前后执行逻辑,比如记录开始时间或处理响应数据。
构建请求执行链
将所有中间件组合成一个调用链。使用数组存储中间件,然后从右到左依次包裹 next 函数:
- 遍历中间件数组,从最后一个开始,逐层封装调用
- 最内层是实际发送 HTTP 请求的 handler
- 最终得到一个可执行的函数,接受初始 config 并返回响应 Promise
例如:
const composed = middlewares.reduceRight((next, mw) => () => mw(config, next), fetchHandler)
封装核心请求方法
提供类似 fetch 或 axios 的简洁 API,如 request、get、post 等。核心 request 方法负责:
- 合并默认配置与用户传入的配置
- 触发中间件链执行
- 统一处理异常并抛出
可以暴露 use 方法用于注册中间件,保持扩展性。
示例:简单实现结构
定义一个 Client 类:
class HttpClient {
constructor() {
this.middlewares = [];
}
use(mw) {
this.middlewares.push(mw);
}
request(config) {
const chain = this.middlewares.reduceRight(
(next, mw) => () => mw(config, next),
() => this._fetch(config)
);
return chain();
}
_fetch(config) {
// 调用浏览器 fetch 或 node 的 http 模块
return fetch(config.url, config);
}
}
这样就能支持按需插入功能模块,比如添加日志中间件:
client.use((config, next) => {
console.log(‘Request:’, config);
return next().then(res => {
console.log(‘Response:’, res);
return res;
});
});
基本上就这些。关键在于把控制流交给中间件链,核心只负责最终发送。结构清晰,易于测试和扩展。不复杂但容易忽略错误冒泡和配置冻结的问题,注意 deep clone 或 immutable 处理即可。
评论(已关闭)
评论已关闭