本文将会你们来看下koa
的源码,固然本文须要你们了解koa
的中间件机制,若是你们以前没有了解过其实现原理,能够关注下这篇文章。koa
的源码很是的精简,与express
不一样,koa
只是为开发者搭起了一个架子,没有任何的功能,包括路由,所有由中间件实现;下面就来看下koa
的实现:node
建立应用时,通常都会利用app.listen
指定一个端口号,这个方法的本质就是http.createServer
:git
listen() { debug('listen'); const server = http.createServer(this.callback()); return server.listen.apply(server, arguments); }
最为关键的就是这个callback
的实现:github
callback() { const fn = compose(this.middleware); if (!this.listeners('error').length) this.on('error', this.onerror); const handleRequest = (req, res) => { res.statusCode = 404; const ctx = this.createContext(req, res); const onerror = err => ctx.onerror(err); const handleResponse = () => respond(ctx); onFinished(res, onerror); return fn(ctx).then(handleResponse).catch(onerror); }; return handleRequest; }
须要注意下面几点:express
onFinished(res, onerror)
,应对的是返回的body
为Stream
的状况,为其添加一个finished
事件。npm
respond()
根据ctx
的status,body,method
来决定如何响应此次请求:api
status
为204,304
,不须要有响应体,res.end()
就好cookie
method
为HEAD
,HEAD
的意义是不请求资源内容可是须要了解资源状况,因此只须要请求头,指定了资源length
后res.end()
就好app
加入body
为空,则body
为statuses
包中status
对应的文字描述,如404 => Not Found
koa
koa
将request
对象response
封装成了一个对象,提供了一些别名,具体能够参见context对象,例如:当访问ctx.url
实则是访问的ctx.request.url
。具体的实现利用了tj写的delegates
这个npm
包来对context
对象添加属性,koa
中利用了其中三个api
:socket
method
:添加方法引用
getter
:利用__defineGetter__
,添加getter
属性
access
:添加getter
与setter
对于context
建立的代码:
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; context.res = request.res = response.res = res; request.ctx = response.ctx = context; request.response = response; response.request = request; context.originalUrl = request.originalUrl = req.url; context.cookies = new Cookies(req, res, { keys: this.keys, secure: request.secure }); request.ip = request.ips[0] || req.socket.remoteAddress || ''; context.accept = request.accept = accepts(req); context.state = {}; return context; }
函数的参数req, res
为node自己的对象,request
和response
分别是对于req
和res
的封装,读取ctx.url
的过程以下:
context
就是一个顶层对象,koa
中,全部的属性和操做基本会基于这个对象,这个对象的组成以下图:
我的感受koa
就像是一个架子,提供了基础的方法和属性,如ctx.redirect
等,具体的功能主要利用中间件来实现,与express
相比,koa
去除内置路由,views
等,变得更加的轻量;固然我认为更加剧要的是避免了层层回调的出现。以上内容若有出错,欢迎你们指出。