摘要:四路由注冊構造函數首先看了解一下構造函數限制必須采用關鍵字服務器支持的請求方法,后續方法會用到保存前置處理函數存儲在構造函數中初始化的和屬性最為重要,前者用來保存前置處理函數,后者用來保存實例化的對象。
一、前言
??Koa為了保持自身的簡潔,并沒有捆綁中間件。但是在實際的開發中,我們需要和形形色色的中間件打交道,本文將要分析的是經常用到的路由中間件 -- koa-router。
??如果你對Koa的原理還不了解的話,可以先查看Koa原理解析。
二、koa-router概述??koa-router的源碼只有兩個文件:router.js和layer.js,分別對應Router對象和Layer對象。
??Layer對象是對單個路由的管理,其中包含的信息有路由路徑(path)、路由請求方法(method)和路由執行函數(middleware),并且提供路由的驗證以及params參數解析的方法。
??相比較Layer對象,Router對象則是對所有注冊路由的統一處理,并且它的API是面向開發者的。
??接下來從以下幾個方面全面解析koa-router的實現原理:
Layer對象的實現
路由注冊
路由匹配
路由執行流程
三、Layer??Layer對象主要是對單個路由的管理,是整個koa-router中最小的處理單元,后續模塊的處理都離不開Layer中的方法,這正是首先介紹Layer的重要原因。
function Layer(path, methods, middleware, opts) { this.opts = opts || {}; // 支持路由別名 this.name = this.opts.name || null; this.methods = []; this.paramNames = []; // 將路由執行函數保存在stack中,支持輸入多個處理函數 this.stack = Array.isArray(middleware) ? middleware : [middleware]; methods.forEach(function(method) { var l = this.methods.push(method.toUpperCase()); // HEAD請求頭部信息與GET一致,這里就一起處理了。 if (this.methods[l-1] === "GET") { this.methods.unshift("HEAD"); } }, this); // 確保類型正確 this.stack.forEach(function(fn) { var type = (typeof fn); if (type !== "function") { throw new Error( methods.toString() + " `" + (this.opts.name || path) +"`: `middleware` " + "must be a function, not `" + type + "`" ); } }, this); this.path = path; // 1、根據路由路徑生成路由正則表達式 // 2、將params參數信息保存在paramNames數組中 this.regexp = pathToRegExp(path, this.paramNames, this.opts); };
??Layer構造函數主要用來初始化路由路徑、路由請求方法數組、路由處理函數數組、路由正則表達式以及params參數信息數組,其中主要采用path-to-regexp方法根據路徑字符串生成正則表達式,通過該正則表達式,可以實現路由的匹配以及params參數的捕獲:
// 驗證路由 Layer.prototype.match = function (path) { return this.regexp.test(path); } // 捕獲params參數 Layer.prototype.captures = function (path) { // 后續會提到 對于路由級別中間件 無需捕獲params if (this.opts.ignoreCaptures) return []; return path.match(this.regexp).slice(1); }
??根據paramNames中的參數信息以及captrues方法,可以獲取到當前路由params參數的鍵值對:
Layer.prototype.params = function (path, captures, existingParams) { var params = existingParams || {}; for (var len = captures.length, i=0; i??需要注意上述代碼中的safeDecodeURIComponent方法,為了避免服務器收到不可預知的請求,對于任何用戶輸入的作為URI部分的內容都需要采用encodeURIComponent進行轉義,否則當用戶輸入的內容中含有"&"、"="、"?"等字符時,會出現預料之外的情況。而當我們獲取URL上的參數時,則需要通過decodeURIComponent進行解碼,而decodeURIComponent只能解碼由encodeURIComponent方法或者類似方法編碼,如果編碼方法不符合要求,decodeURIComponent則會拋出URIError,所以作者在這里對該方法進行了安全化的處理:
function safeDecodeURIComponent(text) { try { return decodeURIComponent(text); } catch (e) { // 編碼方式不符合要求,返回原字符串 return text; } }??Layer還提供了對于單個param前置處理的方法:
Layer.prototype.param = function (param, fn) { var stack = this.stack; var params = this.paramNames; var middleware = function (ctx, next) { return fn.call(this, ctx.params[param], ctx, next); }; middleware.param = param; var names = params.map(function (p) { return p.name; }); var x = names.indexOf(param); if (x > -1) { stack.some(function (fn, i) { if (!fn.param || names.indexOf(fn.param) > x) { // 將單個param前置處理函數插入正確的位置 stack.splice(i, 0, middleware); return true; // 跳出循環 } }); } return this; };??上述代碼中通過some方法尋找單個param處理函數的原因在于以下兩點:
保持param處理函數位于其他路由處理函數的前面;
路由中存在多個param參數,需要保持param處理函數的前后順序。
Layer.prototype.setPrefix = function (prefix) { if (this.path) { this.path = prefix + this.path; // 拼接新的路由路徑 this.paramNames = []; // 根據新的路由路徑字符串生成正則表達式 this.regexp = pathToRegExp(this.path, this.paramNames, this.opts); } return this; };??Layer中的setPrefix方法用于設置路由路徑的前綴,這在嵌套路由的實現中尤其重要。
??最后,Layer還提供了根據路由生成url的方法,主要采用path-to-regexp的compile和parse對路由路徑中的param進行替換,而在拼接query的環節,正如前面所說需要對鍵值對進行繁瑣的encodeURIComponent操作,作者采用了urijs提供的簡潔api進行處理。
四、路由注冊1、Router構造函數
??首先看了解一下Router構造函數:
function Router(opts) { if (!(this instanceof Router)) { // 限制必須采用new關鍵字 return new Router(opts); } this.opts = opts || {}; // 服務器支持的請求方法, 后續allowedMethods方法會用到 this.methods = this.opts.methods || [ "HEAD", "OPTIONS", "GET", "PUT", "PATCH", "POST", "DELETE" ]; this.params = {}; // 保存param前置處理函數 this.stack = []; // 存儲layer };??在構造函數中初始化的params和stack屬性最為重要,前者用來保存param前置處理函數,后者用來保存實例化的Layer對象。并且這兩個屬性與接下來要講的路由注冊息息相關。
??koa-router中提供兩種方式注冊路由:
具體的HTTP動詞注冊方式,例如:router.get("/users", ctx => {})
支持所有的HTTP動詞注冊方式,例如:router.all("/users", ctx => {})
2、http METHODS
??源碼中采用methods模塊獲取HTTP請求方法名,該模塊內部實現主要依賴于http模塊:
http.METHODS && http.METHODS.map(function lowerCaseMethod (method) { return method.toLowerCase() })3、router.verb() and router.all()
??這兩種注冊路由的方式的內部實現基本類似,下面以router.verb()的源碼為例:
methods.forEach(function (method) { Router.prototype[method] = function (name, path, middleware) { var middleware; // 1、處理是否傳入name參數 // 2、middleware參數支持middleware1, middleware2...的形式 if (typeof path === "string" || path instanceof RegExp) { middleware = Array.prototype.slice.call(arguments, 2); } else { middleware = Array.prototype.slice.call(arguments, 1); path = name; name = null; } // 路由注冊的核心處理邏輯 this.register(path, [method], middleware, { name: name }); return this; }; });??該方法第一部分是對傳入參數的處理,對于middleware參數的處理會讓大家聯想到ES6中的rest參數,但是rest參數與arguments其中一個致命的區別:
rest參數只包含那些沒有對應形參的實參,而arguments則包含傳給函數的所有實參。??如果采用rest參數的方式,上述函數則必須要求開發者傳入name參數。但是也可以將name和path參數整合成對象,再結合rest參數:
Router.prototype[method] = function (options, ...middleware) { let { name, path } = options if (typeof options === "string" || options instanceof RegExp) { path = options name = null } // ... return this; };??采用ES6的新特性,代碼變得簡潔多了。
??第二部分是register方法,傳入的method參數的形式就是router.verb()與router.all()的最大區別,在router.verb()中傳入的method是單個方法,后者則是以數組的形式傳入HTTP所有的請求方法,所以對于這兩種注冊方法的實現,本質上是沒有區別的。
4、register
Router.prototype.register = function (path, methods, middleware, opts) { opts = opts || {}; var router = this; var stack = this.stack; // 注冊路由中間件時,允許path為數組 if (Array.isArray(path)) { path.forEach(function (p) { router.register.call(router, p, methods, middleware, opts); }); return this; } // 實例化Layer var route = new Layer(path, methods, middleware, { end: opts.end === false ? opts.end : true, name: opts.name, sensitive: opts.sensitive || this.opts.sensitive || false, strict: opts.strict || this.opts.strict || false, prefix: opts.prefix || this.opts.prefix || "", ignoreCaptures: opts.ignoreCaptures }); // 設置前綴 if (this.opts.prefix) { route.setPrefix(this.opts.prefix); } // 設置param前置處理函數 Object.keys(this.params).forEach(function (param) { route.param(param, this.params[param]); }, this); stack.push(route); return route; };??register方法主要負責實例化Layer對象、更新路由前綴和前置param處理函數,這些操作在Layer中已經提及過,相信大家應該輕車熟路了。
5、use
??熟悉Koa的同學都知道use是用來注冊中間件的方法,相比較Koa中的全局中間件,koa-router的中間件則是路由級別的。
Router.prototype.use = function () { var router = this; var middleware = Array.prototype.slice.call(arguments); var path; // 支持多路徑在于中間件可能作用于多條路由路徑 if (Array.isArray(middleware[0]) && typeof middleware[0][0] === "string") { middleware[0].forEach(function (p) { router.use.apply(router, [p].concat(middleware.slice(1))); }); return this; } // 處理路由路徑參數 var hasPath = typeof middleware[0] === "string"; if (hasPath) { path = middleware.shift(); } middleware.forEach(function (m) { // 嵌套路由 if (m.router) { // 嵌套路由扁平化處理 m.router.stack.forEach(function (nestedLayer) { // 更新嵌套之后的路由路徑 if (path) nestedLayer.setPrefix(path); // 更新掛載到父路由上的路由路徑 if (router.opts.prefix) nestedLayer.setPrefix(router.opts.prefix); router.stack.push(nestedLayer); }); // 不要忘記將父路由上的param前置處理操作 更新到新路由上。 if (router.params) { Object.keys(router.params).forEach(function (key) { m.router.param(key, router.params[key]); }); } } else { // 路由級別中間件 創建一個沒有method的Layer實例 router.register(path || "(.*)", [], m, { end: false, ignoreCaptures: !hasPath }); } }); return this; };??koa-router中間件注冊方法主要完成兩項功能:
將路由嵌套結構扁平化,其中涉及到路由路徑的更新和param前置處理函數的插入;
路由級別中間件通過注冊一個沒有method的Layer實例進行管理。
五、路由匹配Router.prototype.match = function (path, method) { var layers = this.stack; var layer; var matched = { path: [], pathAndMethod: [], route: false }; for (var len = layers.length, i = 0; i < len; i++) { layer = layers[i]; if (layer.match(path)) { // 路由路徑滿足要求 matched.path.push(layer); if (layer.methods.length === 0 || ~layer.methods.indexOf(method)) { // layer.methods.length === 0 該layer為路由級別中間件 // ~layer.methods.indexOf(method) 路由請求方法也被匹配 matched.pathAndMethod.push(layer); // 僅當路由路徑和路由請求方法都被滿足才算是路由被匹配 if (layer.methods.length) matched.route = true; } } } return matched; };??match方法主要通過layer.match方法以及methods屬性對layer進行篩選,返回的matched對象包含以下幾個部分:
path: 保存所有路由路徑被匹配的layer;
pathAndMethod: 在路由路徑被匹配的前提下,保存路由級別中間件和路由請求方法被匹配的layer;
route: 僅當存在路由路徑和路由請求方法都被匹配的layer,才能算是本次路由被匹配上。
??另外,在ES7之前,對于判斷數組是否包含一個元素,都需要通過indexOf方法來實現, 而該方法返回元素的下標,這樣就不得不通過與-1的比較得到布爾值:
if (layer.methods.indexOf(method) > -1) { ... }??而作者巧妙地利用位運算省去了“討厭的-1”,當然在ES7中可以愉快地使用includes方法:
if (layer.methods.includes(method)) { ... }六、路由執行流程??理解koa-router中路由的概念以及路由注冊的方式,接下來就是如何作為一個中間件在koa中執行。
??koa中注冊koa-router中間件的方式如下:
const Koa = require("koa"); const Router = require("koa-router"); const app = new Koa(); const router = new Router(); router.get("/", (ctx, next) => { // ctx.router available }); app .use(router.routes()) .use(router.allowedMethods());??從代碼中可以看出koa-router提供了兩個中間件方法:routes和allowedMethods。
1、allowedMethods()
Router.prototype.allowedMethods = function (options) { options = options || {}; var implemented = this.methods; return function allowedMethods(ctx, next) { return next().then(function() { var allowed = {}; if (!ctx.status || ctx.status === 404) { ctx.matched.forEach(function (route) { route.methods.forEach(function (method) { allowed[method] = method; }); }); var allowedArr = Object.keys(allowed); if (!~implemented.indexOf(ctx.method)) { // 服務器不支持該方法的情況 if (options.throw) { var notImplementedThrowable; if (typeof options.notImplemented === "function") { notImplementedThrowable = options.notImplemented(); } else { notImplementedThrowable = new HttpError.NotImplemented(); } throw notImplementedThrowable; } else { // 響應 501 Not Implemented ctx.status = 501; ctx.set("Allow", allowedArr.join(", ")); } } else if (allowedArr.length) { if (ctx.method === "OPTIONS") { // 獲取服務器對該路由路徑支持的方法集合 ctx.status = 200; ctx.body = ""; ctx.set("Allow", allowedArr.join(", ")); } else if (!allowed[ctx.method]) { if (options.throw) { var notAllowedThrowable; if (typeof options.methodNotAllowed === "function") { notAllowedThrowable = options.methodNotAllowed(); } else { notAllowedThrowable = new HttpError.MethodNotAllowed(); } throw notAllowedThrowable; } else { // 響應 405 Method Not Allowed ctx.status = 405; ctx.set("Allow", allowedArr.join(", ")); } } } } }); }; };??allowedMethods()中間件主要用于處理options請求,響應405和501狀態。上述代碼中的ctx.matched中保存的正是前面matched對象中的path(在routes方法中設置,后面會提到。),在matched對象中的path數組不為空的前提條件下:
服務器不支持當前請求方法,返回501狀態碼;
當前請求方法為OPTIONS,返回200狀態碼;
path中的layer不支持該方法,返回405狀態;
對于上述三種情況,服務器都會設置Allow響應頭,返回該路由路徑上支持的請求方法。
2、routes()
Router.prototype.routes = Router.prototype.middleware = function () { var router = this; // 返回中間件處理函數 var dispatch = function dispatch(ctx, next) { var path = router.opts.routerPath || ctx.routerPath || ctx.path; var matched = router.match(path, ctx.method); var layerChain, layer, i; // 【1】為后續的allowedMethods中間件準備 if (ctx.matched) { ctx.matched.push.apply(ctx.matched, matched.path); } else { ctx.matched = matched.path; } ctx.router = router; // 未匹配路由 直接跳過 if (!matched.route) return next(); var matchedLayers = matched.pathAndMethod var mostSpecificLayer = matchedLayers[matchedLayers.length - 1] ctx._matchedRoute = mostSpecificLayer.path; if (mostSpecificLayer.name) { ctx._matchedRouteName = mostSpecificLayer.name; } layerChain = matchedLayers.reduce(function(memo, layer) { // 【3】路由的前置處理中間件 主要負責將params、路由別名以及捕獲數組屬性掛載在ctx上下文對象中。 memo.push(function(ctx, next) { ctx.captures = layer.captures(path, ctx.captures); ctx.params = layer.params(path, ctx.captures, ctx.params); ctx.routerName = layer.name; return next(); }); return memo.concat(layer.stack); }, []); // 【4】利用koa中間件組織的方式,形成一個‘小洋蔥’模型 return compose(layerChain)(ctx, next); }; // 【2】router屬性用來use方法中區別路由級別中間件 dispatch.router = this; return dispatch; };??routes()中間件主要實現了四大功能。
將matched對象的path屬性掛載在ctx.matched上,提供給后續的allowedMethods中間件使用。(見代碼中的【1】)
將返回的dispatch函數設置router屬性,以便在前面提到的Router.prototype.use方法中區別路由級別中間件和嵌套路由。(見代碼中的【2】)
插入一個新的路由前置處理中間件,將layer解析出來的params對象、路由別名以及捕獲數組掛載在ctx上下文中,這種操作同理Koa在處理請求之前先構建context對象。(見代碼中的【3】)
而對于路由匹配到眾多layer,koa-router通過koa-compose進行處理,這和koa對于中間件處理的方式一樣的,所以koa-router完全就是一個小型洋蔥模型。
七、總結??koa-router雖然是koa的一個中間件,但是其內部也包含眾多的中間件,這些中間件通過Layer對象根據路由路徑的不同進行劃分,使得它們不再像koa的中間件那樣每次請求都執行,而是針對每次請求采用match方法匹配出相應的中間件,再利用koa-compose形成一個中間件執行鏈。
??以上便是koa-router實現原理的全部內容,希望可以幫助你更好的理解koa-router。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/101054.html
摘要:主要通過處理二進制數據流,但是它并不支持字符編碼方式,需要通過模塊進行處理。最后留圖一張往期精彩回顧玩轉原理解析玩轉核心原理分析 一、前置知識 ??在理解koa-bodyparser原理之前,首先需要了解部分HTTP相關的知識。 1、報文主體 ??HTTP報文主要分為請求報文和響應報文,koa-bodyparser主要針對請求報文的處理。 ??請求報文主要由以下三個部分組成: 報文頭...
摘要:云集一線大廠有真正實力的程序員團隊云集一線大廠經驗豐厚的碼農,開源奉獻各教程。融合多種常見的需求場景網絡請求解析模板引擎靜態資源日志記錄錯誤請求處理。結合語句中轉中間件控制權,解決回調地獄問題。注意分支中的目錄為當節課程后的完整代碼。 ?? ?與眾不同的學習方式,為你打開新的編程視角 獨特的『同步學習』方式 文案講解+視頻演示,文字可激發深層的思考、視頻可還原實戰操作過程。 云...
摘要:玩轉同時全面掌握潮流技術采用新一代的開發框架更小更富有表現力更健壯。融合多種常見的需求場景網絡請求解析模板引擎靜態資源日志記錄錯誤請求處理。結合語句中轉中間件控制權,解決回調地獄問題。注意分支中的目錄為當節課程后的完整代碼。 ?? ?與眾不同的學習方式,為你打開新的編程視角 獨特的『同步學習』方式 文案講解+視頻演示,文字可激發深層的思考、視頻可還原實戰操作過程。 云集一線大廠...
POST/GET請求——常見請求方式處理 ?? iKcamp 制作團隊 原創作者:大哼、阿干、三三、小虎、胖子、小哈、DDU、可木、晃晃 文案校對:李益、大力萌、Au、DDU、小溪里、小哈 風采主播:可木、阿干、Au、DDU、小哈 視頻剪輯:小溪里 主站運營:給力xi、xty 教程主編:張利濤 視頻地址:https://www.cctalk.com/v/15114357765870 ...
摘要:詳細代碼如下追蹤賦值里面的是子路由設計子路由設計這個比較簡單,每個子路由維護一個路由監聽列表,然后通過調用的函數添加到主路由列表上。 showImg(https://segmentfault.com/img/bVbruD0?w=756&h=378); 前言 鑒于之前使用express和koa的經驗,這兩天想嘗試構建出一個koa精簡版,利用最少的代碼實現koa和koa-router,同時...
閱讀 2410·2021-10-14 09:43
閱讀 2443·2021-09-09 09:34
閱讀 1606·2019-08-30 12:57
閱讀 1207·2019-08-29 14:16
閱讀 725·2019-08-26 12:13
閱讀 3208·2019-08-26 11:45
閱讀 2290·2019-08-23 16:18
閱讀 2669·2019-08-23 15:27