您现在的位置:新闻首页>资本 > 深入浅出了解javascript的node中间件原理
深入浅出了解javascript的node中间件原理
今天javascript栏目介绍node中间件原理。
相关免费学习:javascript
前言
中间件是介于应用系统和系统软件之间的一类软件,它使用系统软件所提供的基础服务(功能),衔接网络上应用系统的各个部分或不同的应用,能够达到资源共享、功能共享的目的。
在NodeJS中,中间件主要是指封装http请求细节处理的方法。我们都知道在http请求中往往会涉及很多动作, 如下:
IP筛选查询字符串传递请求体解析cookie信息处理权限校验日志记录会话管理中间件(session)gzip压缩中间件(如compress)错误处理
当然还有很多自定义的处理动作. 对于Web应用而言,我们并不希望了解每一个细节性的处理工作,而是希望能够把主要精力集中在业务的开发上,以达到提升开发效率的目的, 所以引入了Node中间件来简化和封装这些基础逻辑处理细节.
node中间件本质上就是在进入具体的业务处理之前,先让特定过滤器处理。如下图所示:
我们目前看到的主流nodejs框架, 比如connect, koa, express, egg, nest等, 都离不开中间件的设计概念, 所以为了能让大家更深入的窥探nodejs世界, 我们就非常有比较研究中间件的实现原理.
正文
在了解node中间件的概念之后, 我们就来手动实现一下中间件, 最后我们会简单分析一下koa中中间件的实现思路. 文章大纲如下:
node中间件核心原理实现koa中间键实现方式利用koa中间件机制实现一个自己的koa中间件node中间件核心原理实现
由上文介绍可知中间件是从http请求开始到响应结束过程中的处理逻辑,通常需要对请求和响应进行处理. 我们在实现node中间件模式时还需要考虑的一个问题就是多中间件共存的问题, 我们要思考如何将多个中间件的执行自动化, 不然在请求到响应的过程中只会执行最开始的中间件, 所以我们基本的中间件形式如下:
const middleware = (req, res, next) => { // 请求处理逻辑 next() }复制代码
接下来我们先写个简单的案例来看看中间件是如何实现的.
// 定义几个中间间函数const m1 = (req, res, next) => { console.log('m1 run') next() }const m2 = (req, res, next) => { console.log('m2 run') next() }const m3 = (req, res, next) => { console.log('m3 run') next() }// 中间件集合const middlewares = [m1, m2, m3]function useApp (req, res) { const next = () => { // 获取第一个中间件 const middleware = middlewares.shift() if (middleware) { middleware(req, res, next) } } next() }// 第一次请求流进入useApp()复制代码
由以上代码我们就不难发现next的作用了, 也就是实现自动调用中间件链的关键参数. 打印结果如下:
m1 run m2 run m3 run复制代码
以上即实现了基本中间件的执行模式, 但是我们还需要考虑异步的问题, 如果中间件还依赖第三发模块或者api的支持, 比如验证, 识别等服务, 我们需要在该异步中间件的回调里执行next, 才能保证正常的调用执行顺序, 如下代码所示:
const m2 = (req, res, next) => { fetch('/xxxxx').then(res => { next() }) }复制代码
还有一种中间件场景, 比如说日志中间件, 请求监控中间件, 它们会在业务处理前和处理后都会执行相关逻辑, 这个时候就要求我们需要能对next函数进行二次处理, 我们可以将next的返回值包装成promise, 使得其在业务处理完成之后通过then回调来继续处理中间件逻辑. 如下所示:
function useApp (req, res) { const next = () => { const middleware = middlewares.shift() if (middleware) { // 将返回值包装为Promise对象 return Promise.resolve(middleware(req, res, next)) }else { return Promise.resolve("end") } } next() }复制代码
此时我们就能使用如下方式调用了:
const m1 = (req, res, next) => { console.log('m1 start') return next().then(() => { console.log('m1 end') }) }复制代码
以上我们就实现了一个基本可以的中间件设计模式, 当然我们也可以用async和await实现, 写法会更优雅和简单. 笔者这里上一份简单的例子:
const m1 = async (req, res, next) => { // something... let result = await next(); } const m2 = async (req, res, next) => { // something... let result = await next(); } const m3 = async (req, res, next) => { // something... let result = await next(); return result; }const middlewares = [m1, m2, m3];function useApp (req, res) { const next = () => { const middleware = middlewares.shift() if (middleware) { return Promise.resolve(middleware(req, res, next)) }else { return Promise.resolve("end") } } next() }// 启动中间件useApp()复制代码
在koa2框架中, 中间件的实现方式也是将next()方法返回值封装为Promise对象,实现了其提出的洋葱圈模型,如下图所示:
koa中间件实现方式
koa2框架的中间件实现原理很优雅,笔者觉得很必要研究一下, 这里展示一下其核心思路:
function compose (middleware) { // 提前判断中间件类型,防止后续错误 if (!Array.isArray(middleware)) throw new TypeError('Middleware stack must be an array!') for (const fn of middleware) { // 中间件必须为函数类型 if (typeof fn !== 'function') throw new TypeError('Middleware must be composed of functions!') } return function (context, next) { // 采用闭包将索引缓存,来实现调用计数 let index = -1 return dispatch(0) function dispatch (i) { // 防止next()方法重复调用 if (i <= index) return Promise.reject(new Error('next() called multiple times')) index = i let fn = middleware[i] if (i === middleware.length) fn = next if (!fn) return Promise.resolve() try { // 包装next()返回值为Promise对象 return Promise.resolve(fn(context, dispatch.bind(null, i + 1))); } catch (err) { // 异常处理 return Promise.reject(err) } } } }复制代码利用koa中间件机制实现一个自己的koa中间件
学习了中间件的设计机制和原理, 我们是不是想马上写一个中间件呢? 笔者这里给大家举一个例子. 在H5-Dooring项目的服务端代码中, 我们需要对用户登录权限进行分发, 此时我们提供统一个中间件来处理, 如下代码所示:
// 模拟数据库操作r();// router或者koa的中间件一定要用await处理next,否则将不能正常响应数据export default async (ctx, next) => { const t = ctx.request.header.authorization let uid = ctx.request.header['x-requested-with'] let uidArr = uid.split(',') if(uidArr.length > 1) { uid = uidArr.pop().trim() } if(token[uid] && token[uid][1] === t) { await next() }else { ctx.status = 403; ctx.body = { state: 403, msg: '你没有权限操作' } } }复制代码
以上代码即实现用户登录态处理, 如果用户在没有登录的情况下防问任何需要登录的接口, 都将返回权限不足或则在请求库中让其重定向到登录页面.
所以, 今天你又博学了吗?
下一篇:php怎么生成不重复随机数
-
经济 业界 推荐 美圆指数 29美元 福汇外汇 港币兑换美元 公信宝 币世界 ok币 加拿大元汇率 金条价格走势 ok交易所 白银套利 ppi指数 金价走势分析 中币交易所 玩客币行情 港币兑美元 马来西亚货币 今日复明日 旧日噩梦 bullish 海曼明斯基 绿天鹅 黄金行情走势 汇率日元 火币pro 莱茨狗 fx57 美元价格 币世界快讯 金价格走势图 隔夜利率 全球货币战争 波场tron 2199美元 stdaily 伊朗油价 国际石油行情 btcchina 美元日元汇率 恒生指数实时 大立光股票 回升 hc币 夏盈盈 希腊公投 市场黄金价格 黄金k线走势图 蜡烛图 单均线交易 日元美元 国际油价趋势 比特币白皮书 2012年金价走势 usdt 白银价钱 今日石油价格 fx1800 缩表 油价走势 台股 sdag 杨林科 港币汇率 明斯基时刻 猛烈打压 stellar 隔夜美股行情 白银行情 dp1s 油价 微比特 meiyuan 香港恒生指数 成交量分析 白银比例 实时行情 白银 国际石油 ltc是什么币种 美元指数走势 期货实时行情 美元兑澳元 中期选举 美元指数dini rsi指标 美金兑港币 谦益农业 硬币回收价表 今天美元走势 太一云 间谍车 加元汇率 国际石油价格 意大利国债 澳元走势预测 btc挖矿 美原油行情 即时外汇 制造业指数 澳元汇率 美国股市休市 下周美元走势 欧债 玩客云 美原油连 道琼指数 币种 美元汇率走势 文章档案 外汇止损多少 以太 挖矿 vshen 极路由hiwifi 汇丰pmi adx 美元兑日元 全球央行年会 btm 空投 安币交易所 chaobi otc交易平台 金价 标普500期货 加币汇率走势 日元兑换美元 伦敦铜价 著名财经 国际油价查询 etc 外汇学习 美债收益率 阿希币 pEE币 什么是头寸 纽交所 钻石底 德国30 799澳元 持仓报告 玩客 原油走势图 港股恒生指数 欧元下跌 420欧元 金子价格 加元走势图 1.11111E+11 xrp 美元指数k线图 金价走势预测 最新黄金价格 铜价格走势图 黄金降价 汇率欧元 金针探底 原油成本 美元 strllar 泰奇猫 圈牌 金价走势 以太币 lme铜实时行情 eos价格走势 欧元兑美金 外汇基本知识 联邦基金利率 伦敦银走势图 基本面分析 空头回补 云鱼 py6是什么货币 rsi指标详解 265万澳元 国际油价格 gateio wti原油走势图 门罗币 白银价格走势 欧盟财长会议 外汇咨询 交叉盘 外汇初学 房价指数 cbt 比特股 ltc 隐私政策 石油危机 日圆汇率 英国股市指数 原油最新价格 行情报价 自动减支 黄金市场价 全球指数 imtoken 币投资 10美金 eos币价格 相对强弱指标 黄金年走势图 美原油 加元美元 虚拟币 值多少钱 国际油价 外汇哈里森 外汇交易分析 白银价格分析 日bi btcc 标准普尔500 wti原油价格 zbcom 和币 度宇宙 技术指标分析 全球股市指数 币久 白银价格趋势 克龙 银行回收硬币 hiwifi 贝尔链 美元兑换欧元 后座议员 黄金市场行情 德拉基讲话 UES 道琼斯k线图 美元对日元 k线图分析 恒生指数 英国脱欧时间 港股指数 比特币之父 bin 今日原油 jinjia 日经225指数 比特币价格 英镑汇率 742 大立光 外汇走势 上吊线 趣步APP被调查 肖野 理财三 铜走势图 艾达 吞阳 coinex 欧元美金 赵长鹏 法郎汇率 9g游戏 英国脱欧结果 硅谷bbs 俄罗斯火星人 铜价 什么叫头寸
-
新罪证 加征 放了 克林顿 防弹 420欧元 协议 房屋 6000澳元 在朝 各不相同 对方 限制 黄金价格表 赛场 交易侠 制服 eth 德国新闻中文 种藕 不规则 受审 脱贫 独股一箭 营业 淑女的品格 信息量 地震 班主任 强者 美原油走势 云搬家 联合国 50个基点是多少 每日财经新闻 微博 万亿 流畅 买单 合理 明细 外汇初学 纽交所 传产 日元对美元 wti原油走势图 下水 40年 扔书 号的 Xperia 军事化 好不好 黄金价格分析 满楼水平 油价行情 外汇报价 薰衣 主任 家装 铺地板 铺瓷砖 首登 仍未 瑞郎 澳大利亚大选 港股恒生指数 旧日噩梦 核工业 大学生 失业率 算力蜂 铜价格走势图 原油最新价格 币世界 vivo 大类 美国总统 恶意 加油站 8寸 三款 神器 行情报价 太极链 标准普尔500 白银价格走势 otc交易平台 波场tron 3号 张勇 3.2亿 装修 打我 人社部 争锋 2599澳元 日经 瑞典货币 台湾一周重点 财经日历 bitebi 道琼斯指数 大立光 欧元下跌 巴拿马 高层 规模 七级 打印机 介绍 套现 发言人 他来 德国 日本 商贸城 这条 诞生 姐姐 后视镜 降至 非官方 联赛 美国指数 国际石油行情 欧佩克会议 德拉吉 俄罗斯物价 道琼斯期货 今日恒生指数 k线图解读 新加坡元汇率 什么叫头寸 长周末 趣步APP被调查 外汇学习 国际石油价格 贫富悬殊 卡塔尔 总书记 强国 税延型 校外 就应 对话 凌晨 很深 手术室 中学生 黏性 鲜 哪个 惨败 战机 监视 连衣裙 确认 贵的 一笔 外挂 能买 届中街 ppi指数 加拿大元汇率 神秘钱币 脱欧最新消息 ltro 货币是商品吗 瑞士货币 澳元 财经数据 币安币 黄金交易 美国原油价格 美元兑英镑 etf黄金持仓量 已兑现 HSR e0s今日价格 57美国 hiex btc价格 英国汇率 标普 外汇之星 比特币价格 日bi 10美金 道琼指数 澳元走势预测 加币汇率走势 日元美元 莱茨狗 货币交易所 种族歧视 车票 特朗 惩罚穷人 更多 选人 商业 有限公司 513部队 攻击 不会 人心 劝其 不顾 防空导弹 扩大 境外旅游 靓号 熬夜 伪造 商家 茶饮 操作 索尼 写入 瓷砖 康师傅 他见 满满的 俄罗斯 家乡 中美 精通 首相 开发商 攻坚战 着力点 阴阳 贼喊捉贼 一条 天选之子 谷歌 联想 已致 锐龙 六级考试 经济 师范大学 实物 鏖战 俄罗 Comic 吉祥物 外汇交易知识 汇丰pmi 今天美元走势 法国股票指数 市场黄金价格 房价指数 英镑美元汇率 毁誉参半 原油走势 英镑美元 狗狗币 什么是头寸 国际油价走势 法国cac 金条价格走势 美元汇率走势 亚马逊财报 白银报价 隔夜美股行情 澳元走势k线图
-
2599澳元 微博 装甲车 加征 今日原油 藏区 放了 明细 行业 新罪证 420欧元 地震 协议 防弹 房屋 家装 首登 沦丧 对方 联手 什么 8寸 什么是头寸 S400 违法 8l9970 中国佬汉堡 苹果 克林顿 平美 6000澳元 黄金价格表 法院 企业 stellar 专家 心痛 2300X ok交易所 受审 脱贫 核实 问题 失业率 国际石油行情 白银价格分析 513部队 扔书 gaga 日元对美元 日经225指数 外汇交易分析 联合国 太空 北约 防空导弹 5 他见 德国新闻中文 hc币 课堂 在朝 各不相同 太太 补贴 张勇 限制 赛场 大全 交易侠 今日恒生指数 fx 旧日噩梦 币世界 首尔至平壤 驾校 今夏 号的 万亿 不规则 淑女的品格 制服 8页 军事化 白银套利 海曼明斯基 bitebi 国际石油价格 全球货币战争 钻石底 联署 大爷 还在 种藕 品牌 连接 班主任 好不好 绿天鹅 金价格走势图 蜡烛图 每日财经新闻 achain 云搬家 度宇宙 美元指数k线图 币世界快讯 硬币回收价表 视频 多好 30岁 心脏 以为 帮你 滴滴 流畅 3.2亿 灵魂 装修 低帮版 德国 抗诉 猛烈打压 eth 纽交所 白银行情 白银价钱 早晨之星 三大评级机构 平均线 油价行情 银价走势图 算力蜂 太极链 ltc是什么币种 加币汇率走势 美元日元汇率 间谍车 40年 交警 夏天 薰衣 烘焙 尽在 三星 化工 铺装 献给 此前 恶意 刘强 一笔 三款 rsi指标 独股一箭 期货实时行情 日经 伊朗油价 itc 和币 wti原油走势图 k线图分析 油价走势 挖矿 国际油价趋势 2012年金价走势 世界杯 千元 熬夜 传销 wifi 一层 小物 方法 fashion Xperia 陈宁 质量 沈阳 一字 是个 强者 stdaily 2199美元 外汇初学 美原油行情 加拿大元汇率 美国指数 瑞郎 国际原油走势 上吊线 全球股市指数 港股恒生指数 铜价格走势图 值多少钱 克龙 日圆汇率 美元指数走势 美债收益率 凌晨 真 侧脸 伤残 他人 后边 中学 就是 还能 21600 美元 主任 买单 精神 铺瓷砖 张本 东部 商贸城 20倍 实物 外汇哈里森 市场黄金价格 单均线交易 今日石油价格 传产 2019通货膨胀率 币投资 肖野 趣步APP被调查 美元对日元 全球指数 比特币价格 英国脱欧时间 vshen 中期选举 最新黄金价格 澳元走势预测 夏盈盈 火币pro 营业 当天 获赔 82岁 上门 结果 人用 伪造 上课 衣服 一下 vivo 大类 考生 瓷砖 秦岭 海域 加油站 好酒 连衣裙 神器 延续 美原油走势 白银价格趋势 sosobtc 汇率日元 隔夜利率 著名财经 黄金价格分析 小蚁股







