玩转Koa之核心原理分析


Posted in Javascript onDecember 29, 2018

Koa作为下一代Web开发框架,不仅让我们体验到了async/await语法带来同步方式书写异步代码的酸爽,而且本身简洁的特点,更加利于开发者结合业务本身进行扩展。

本文从以下几个方面解读Koa源码:

  • 封装创建应用程序函数
  • 扩展res和req
  • 中间件实现原理
  • 异常处理

 一、封装创建应用程序函数

利用NodeJS可以很容易编写一个简单的应用程序:

const http = require('http')

const server = http.createServer((req, res) => {
 // 每一次请求处理的方法
 console.log(req.url)
 res.writeHead(200, { 'Content-Type': 'text/plain' })
 res.end('Hello NodeJS')
})

server.listen(8080)

注意:当浏览器发送请求时,会附带请求/favicon.ico。

而Koa在封装创建应用程序的方法中主要执行了以下流程:

  • 组织中间件(监听请求之前)
  • 生成context上下文对象
  • 执行中间件
  • 执行默认响应方法或者异常处理方法
// application.js
listen(...args) {
 const server = http.createServer(this.callback());
 return server.listen(...args);
}

callback() {
 // 组织中间件
 const fn = compose(this.middleware);

 // 未监听异常处理,则采用默认的异常处理方法
 if (!this.listenerCount('error')) this.on('error', this.onerror);

 const handleRequest = (req, res) => {
  // 生成context上下文对象
  const ctx = this.createContext(req, res);
  return this.handleRequest(ctx, fn);
 };

 return handleRequest;
}

handleRequest(ctx, fnMiddleware) {
 const res = ctx.res;
 // 默认状态码为404
 res.statusCode = 404;
 // 中间件执行完毕之后 采用默认的 错误 与 成功 的处理方式
 const onerror = err => ctx.onerror(err);
 const handleResponse = () => respond(ctx);
 onFinished(res, onerror);
 return fnMiddleware(ctx).then(handleResponse).catch(onerror);
}

二、扩展res和req

首先我们要知道NodeJS中的res和req是http.IncomingMessage和http.ServerResponse的实例,那么就可以在NodeJS中这样扩展req和res:

Object.defineProperties(http.IncomingMessage.prototype, {
 query: {
  get () {
   return querystring.parse(url.parse(this.url).query)
  }
 }
})

Object.defineProperties(http.ServerResponse.prototype, {
 json: {
  value: function (obj) {
   if (typeof obj === 'object') {
    obj = JSON.stringify(obj)
   }
   this.end(obj)
  }
 }
})

而Koa中则是自定义request和response对象,然后保持对res和req的引用,最后通过getter和setter方法实现扩展。

// application.js
createContext(req, res) {
 const context = Object.create(this.context);
  const request = context.request = Object.create(this.request);
  const response = context.response = Object.create(this.response);
  context.app = request.app = response.app = this;
  context.req = request.req = response.req = req; // 保存原生req对象
  context.res = request.res = response.res = res; // 保存原生res对象
  request.ctx = response.ctx = context;
  request.response = response;
  response.request = request;
  context.originalUrl = request.originalUrl = req.url;
  context.state = {};
  // 最终返回完整的context上下文对象
  return context;
}

所以在Koa中要区别这两组对象:

  • request、response: Koa扩展的对象
  • res、req: NodeJS原生对象
// request.js
get header() {
 return this.req.headers;
},
set header(val) {
 this.req.headers = val;
},

此时已经可以采用这样的方式访问header属性:

ctx.request.header

但是为了方便开发者调用这些属性和方法,Koa将response和request中的属性和方法代理到context上。

通过Object.defineProperty可以轻松的实现属性的代理:

function access (proto, target, name) {
 Object.defineProperty(proto, name, {
  get () {
   return target[name]
  },
  set (value) {
   target[name] = value
  }
 })
}

access(context, request, 'header')

而对于方法的代理,则需要注意this的指向:

function method (proto, target, name) {
 proto[name] = function () {
  return target[name].apply(target, arguments)
 }
}

上述就是属性代理和方法代理的核心代码,这基本算是一个常用的套路。

代理这部分详细的源码,可以查看node-delegates , 不过这个包时间久远,有一些老方法已经废除。

在上述过程的源码中涉及到很多JavaScript的基础知识,例如:原型继承、this的指向。对于基础薄弱的同学,还需要先弄懂这些基础知识。

三、中间件实现原理

首先需要明确是:中间件并不是NodeJS中的概念,它只是connect、express和koa框架衍生的概念。

1、connect中间件的设计

在connect中,开发者可以通过use方法注册中间件:

function use(route, fn) {
 var handle = fn;
 var path = route;

 // 不传入route则默认为'/',这种基本是框架处理参数的一种套路
 if (typeof route !== 'string') {
  handle = route;
  path = '/';
 }

 ...
 // 存储中间件
 this.stack.push({ route: path, handle: handle });
 
 // 以便链式调用
 return this;
}

use方法内部获取到中间件的路由信息(默认为'/')和中间件的处理函数之后,构建成layer对象,然后将其存储在一个队列当中,也就是上述代码中的stack。

connect中间件的执行流程主要由handle与call函数决定:

function handle(req, res, out) {
 var index = 0;
 var stack = this.stack;
 ...
 function next(err) {
  ...
  // 依次取出中间件
  var layer = stack[index++]

  // 终止条件
  if (!layer) {
   defer(done, err);
   return;
  }

  var path = parseUrl(req).pathname || '/';
  var route = layer.route;

  // 路由匹配规则
  if (path.toLowerCase().substr(0, route.length) !== route.toLowerCase()) {
   return next(err);
  }
  ...
  call(layer.handle, route, err, req, res, next);
 }

 next();
}

handle函数中使用闭包函数next来检测layer是否与当前路由相匹配,匹配则执行该layer上的中间件函数,否则继续检查下一个layer。

这里需要注意next中检查路由的方式可能与想象中的不太一样,所以默认路由为'/'的中间件会在每一次请求处理中都执行。

function call(handle, route, err, req, res, next) {
 var arity = handle.length;
 var error = err;
 var hasError = Boolean(err);

 try {
  if (hasError && arity === 4) {
   // 错误处理中间件
   handle(err, req, res, next);
   return;
  } else if (!hasError && arity < 4) {
   // 请求处理中间件
   handle(req, res, next);
   return;
  }
 } catch (e) {
  // 记录错误
  error = e;
 }

 // 将错误传递下去
 next(error);
}

在通过call方法执行中间件方法的时候,采用try/catch捕获错误,这里有一个特别需要注意的地方是,call内部会根据是否存在错误以及中间件函数的参数决定是否执行错误处理中间件。并且一旦捕获到错误,next方法会将错误传递下去,所以接下来普通的请求处理中间件即使通过了next中的路由匹配,仍然会被call方法给过滤掉。

下面是layer的处理流程图:

玩转Koa之核心原理分析

上述就是connect中间件设计的核心要点,总结起来有如下几点:

  • 通过use方法注册中间件;
  • 中间件的顺序执行是通过next方法衔接的并且需要手动调用,在next中会进行路由匹配,从而过滤掉部分中间件;
  • 当中间件的执行过程中发生异常,则next会携带异常过滤掉非错误处理中间件,也是为什么错误中间件会比其他中间件多一个error参数;
  • 在请求处理的周期中,需要手动调用res.end()来结束响应;

 2、Koa中间件的设计

Koa中间件与connect中间件的设计有很大的差异:

  • Koa中间件的执行并不需要匹配路由,所以注册的中间件每一次请求都会执行。(当然还是需要手动调用next);
  • Koa中通过继承event,暴露error事件让开发者自定义异常处理;
  • Koa中res.end由中间件执行完成之后自动调用,这样避免在connect忘记调用res.end导致用户得不到任何反馈。
  • Koa中采用了async/await语法让开发者利用同步的方式编写异步代码。

当然,Koa中也是采用use方法注册中间件,相比较connect省去路由匹配的处理,就显得很简洁:

use(fn) {
 this.middleware.push(fn);
 return this;
}

并且use支持链式调用。

Koa中间件的执行流程主要通过koa-compose中的compose函数完成:

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!')
 }

 /**
  * @param {Object} context
  * @return {Promise}
  * @api public
  */

 return function (context, next) {
  let index = -1
  return dispatch(0)
  function dispatch (i) {
   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 {
    // 递归调用下一个中间件
    return Promise.resolve(fn(context, dispatch.bind(null, i + 1))); 
   } catch (err) {
    return Promise.reject(err)
   }
  }
 }
}

看到这里本质上connect与koa实现中间件的思想都是递归,不难看出koa相比较connect实现得更加简洁,主要原因在于:

  • connect中提供路由匹配的功能,而Koa中则是相当于connect中默认的'/'路径。
  • connect在捕获中间件的异常时,通过next携带error一个个中间件验证,直到错误处理中间件,而Koa中则是用Promise包装中间件,一旦中间件发生异常,那么会直接触发reject状态,直接在Promise的catch中处理就行。

上述就是connect中间件与Koa中间件的实现原理,现在在再看Koa中间件的这张执行流程图,应该没有什么疑问了吧?!

四、异常处理

对于同步代码,通过try/catch可以轻松的捕获异常,在connect中间件的异常捕获则是通过try/catch完成。

对于异步代码,try/catch则无法捕获,这时候一般可以构造Promise链,在最后的catch方法中捕获错误,Koa就是这样处理,并且在catch方法中发送error事件,以便开发者自定义异常处理逻辑。

this.app.emit('error', err, this);

前面也谈到Koa利用async/await语法带来同步方式书写异步代码的酸爽,另外也让错误处理更加自然:

// 也可以这样自定义错误处理
app.use(async (ctx, next) => {
 try {
  await next();
 } catch (err) {
  ctx.status = err.status || 500
  ctx.body = err
 }
})

五、总结

相信看到这里,再回忆一下之前遇到的那些问题,你应该会有新的理解,并且再次使用Koa时会更加得心应手,这也是分析Koa源码的目的之一。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
防止页面被iframe(兼容IE,Firefox火狐)
Jul 04 Javascript
JavaScript基础教程之alert弹出提示框实例
Oct 16 Javascript
javascript实现相同事件名称,不同命名空间的调用方法
Jun 26 Javascript
JQuery中Ajax()的data参数类型实例分析
Dec 15 Javascript
浅谈JS运算符&amp;&amp;和|| 及其优先级
Aug 10 Javascript
Query常用DIV操作获取和设置长度宽度的实现方法
Sep 19 Javascript
JS简单实现数组去重的方法示例
Mar 27 Javascript
详解JS中的this、apply、call、bind(经典面试题)
Sep 19 Javascript
jQuery中过滤器的基本用法示例
Oct 11 jQuery
解决layui中的form表单与button的点击事件冲突问题
Aug 15 Javascript
JS实现简单的抽奖转盘效果示例
Feb 16 Javascript
vuex(vue状态管理)的特殊应用案例分享
Mar 03 Javascript
如何从0开始用node写一个自己的命令行程序
Dec 29 #Javascript
bootstrap与pagehelper实现分页效果
Dec 29 #Javascript
微信小程序实现滑动切换自定义页码的方法分析
Dec 29 #Javascript
vue实现分页组件
Jun 16 #Javascript
如何用RxJS实现Redux Form
Dec 29 #Javascript
Vue.js结合bootstrap前端实现分页和排序效果
Dec 29 #Javascript
jQuery实现的鼠标拖动浮层功能示例【拖动div等任何标签】
Dec 29 #jQuery
You might like
《星际争霸II》全新指挥官斯台特曼现已上线
2020/03/08 星际争霸
Protoss兵种对照表
2020/03/14 星际争霸
PHP批量删除、清除UTF-8文件BOM头的代码实例
2014/04/14 PHP
ThinkPHP5.0 图片上传生成缩略图实例代码说明
2018/06/20 PHP
用正则获取指定路径文件的名称
2007/02/27 Javascript
jQuery学习笔记(3)--用jquery(插件)实现多选项卡功能
2013/04/08 Javascript
javascript十六进制及二进制转化的方法
2015/05/06 Javascript
jQuery地图map悬停显示省市代码分享
2015/08/20 Javascript
js密码强度检测
2016/01/07 Javascript
详解javascript事件冒泡
2016/01/09 Javascript
详解angularjs利用ui-route异步加载组件
2017/05/21 Javascript
微信小程序后台解密用户数据实例详解
2017/06/28 Javascript
基于vue,vue-router, vuex及addRoutes进行权限控制问题
2018/05/02 Javascript
基于JS实现web端录音与播放功能
2019/04/17 Javascript
详解基于Vue的支持数据双向绑定的select组件
2019/09/02 Javascript
[10:21]2018DOTA2国际邀请赛寻真——Winstrike
2018/08/11 DOTA
python网络编程学习笔记(二):socket建立网络客户端
2014/06/09 Python
python获取android设备的GPS信息脚本分享
2015/03/06 Python
Django中模型Model添加JSON类型字段的方法
2015/06/17 Python
Python实现自动上京东抢手机
2018/02/06 Python
Python获取系统所有进程PID及进程名称的方法示例
2018/05/24 Python
Python函数中参数是传递值还是引用详解
2019/07/02 Python
Spring Boot中使用IntelliJ IDEA插件EasyCode一键生成代码详细方法
2020/03/20 Python
Opencv图像处理:如何判断图片里某个颜色值占的比例
2020/06/03 Python
python实现梯度下降算法的实例详解
2020/08/17 Python
Python+unittest+requests+excel实现接口自动化测试框架
2020/12/23 Python
德国价格合理的品牌商品购物网站:averdo
2019/03/21 全球购物
M.M.LaFleur官网:美国职业女装品牌
2020/10/27 全球购物
怎样写留学自荐信
2013/11/11 职场文书
股权转让协议书
2014/04/12 职场文书
政府领导干部个人对照检查材料思想汇报
2014/09/24 职场文书
群众路线自我剖析及整改措施
2014/11/04 职场文书
2014年除四害工作总结
2014/12/06 职场文书
一起来学习Python的元组和列表
2022/03/13 Python
TV动画《史上最强大魔王转生为村民A》番宣CM公布
2022/04/01 日漫
Python实现双向链表基本操作
2022/05/25 Python