route.js并不是exPRess里真正的路由代码,他只是其中的一个组成部分,和router(router/index.js)是有区别的。下面先看一下重要的代码。
function Route(path) { this.path = path; this.stack = []; // route handlers for various http methods this.methods = {};}Route.prototype.dispatch = function dispatch(req, res, done) {//done是router的next函数,执行后会跳到下一个中间件,从而跳过当前route的stack中剩下的函数 var idx = 0; var stack = this.stack; if (stack.length === 0) { return done();//done是开始执行该路由组件时传进来的回调,执行完路由栈里的函数后执行done } var method = req.method.toLowerCase(); if (method === 'head' && !this.methods['head']) { method = 'get'; } req.route = this; next(); function next(err) { if (err && err === 'route') {//使用next('route')会不执行剩下的函数 return done(); } var layer = stack[idx++]; if (!layer) {//已经遍历完layer了,跳到下一个router中的layer return done(err); } if (layer.method && layer.method !== method) {//方法不匹配,next return next(err); } if (err) {//通过层层筛选,最后可以执行的路由函数,先判断有没有错误,再执行相应的函数,每个layer都有对应的回调。 layer.handle_error(err, req, res, next); } else { layer.handle_request(req, res, next); } }};Route.prototype.all = function all() { var handles = flatten(slice.call(arguments)); for (var i = 0; i < handles.length; i++) { var handle = handles[i]; if (typeof handle !== 'function') { var type = toString.call(handle); var msg = 'Route.all() requires callback functions but got a ' + type; throw new TypeError(msg); } var layer = Layer('/', {}, handle); layer.method = undefined; this.methods._all = true; this.stack.push(layer); } return this;};首先route是作为一个layer对象的一个属性,挂在在layer中的,route本身的结果类似于router,是一个数组,或者说是一个栈。元素的类型是layer。 1.构造函数比较简单,主要是存储路由的路径和分配一个数组存储layer数据结构。methods的存在主要是因为前端请求一个路径,可以对应很多种http方法,比如get,post这种,所以methods变量主要是用在匹配路由时进行筛选时。 2.各种http方法和一个特殊的all方法的逻辑是类似的,主要是根据传进来的0个或者多个回调函数,创建0个或者多个layer层,并且把他们加到route的数组中。 3.dispatch是最重要的函数,这个函数是在路由查找的过程中,匹配到了当前的路由时执行。核心的逻辑在next函数里,并且这个next函数和router里的next是不一样的。代码的第一行先判断执行next的时候,是否传入了route,是的话,就忽略当前栈中还没有执行的函数,跳到下一个路由层。我们可以想象express的整个路由机制有点类似二维数组,比如是5*5的,我们先在一维数组里查找路由(router的逻辑),找到后,假设是第二行,然后我们执行第二行里的所有列对应的函数(route的逻辑)。回到上面,假设是在第二列next到第三列时,执行next的时候传入了route,那么第三列和后面的函数都不会被执行,路由机制会继续从第三行往后找匹配的路径。这里是通过执行done达到这个效果。
新闻热点
疑难解答