generator探幽(1)--koa中间件机制浅析
本系列旨在通過對co,koa等庫源碼的研究,進而理解generator在異步編程中的重大作用(ps:所有代碼請在node --harmony或者iojs環境中運行)
koa中間件的形式
相信用過koa的小伙伴一定很熟悉下面這段代碼
var app = require('koa')(),router = require('koa-router')();app.use(function *(next){console.log(1);yield next;console.log(5); }); app.use(function *(next){console.log(2);yield next;console.log(4); }); app.use(function *(){console.log(3); });app.listen(3000);當一個請求到達的時候,控制臺會依次輸出1 2 3 4 5,這就是koa中強大的middleware特性,撇開koa本身,我們來扯扯middleware這東西是怎么實現的
yield *iterator
不熟悉generator和iterator的小伙伴可以先看下阮一峰先生編寫的《ES6入門》的generator函數這一節
假設你已經熟悉了generator和iterator,現在我們來看一段代碼
上面代碼運行后,控制臺會依次輸出1 2 3,這是因為在m1函數內部,yield后面跟的是一個iterator。
m1Iterator.next()調用后,m1函數開始執行。
當m1運行到yield *m2Iterator時,不會暫停,而是直接跳進m2函數執行m2函數內的代碼。
由于m2函數中沒有yield,因此會一直執行完m2函數中的代碼,并返回至m1函數中執行yield *m2Iterator后面的代碼。
middleware中間件的雛形
理解了上面的過程后,我們接著看下面這段代碼
function *m1(){console.log(1);yield *m2();console.log(5); }function *m2(){console.log(2);yield *m3();console.log(4); }function *m3(){console.log(3); }m1().next(); //控制臺依次輸出1 2 3 4 5是不是形式跟我們一開始看到的koa的中間件有點接近了?
compose
compose是koa里用來實現中間件機制的模塊
compose函數最關鍵的一段代碼
這段代碼把middleware中傳進來的generator數組逆序取出并依次執行,每次執行的時候把上次generator執行返回的iterator當作參數傳進去,所有generator執行完之后,調用第一個iterator
因此,在generator中間件函數中,其實yield后面跟的是個iterator,即yield *next
下面用一個例子來說明這個過程
現在就很像koa的中間件的寫法了。唯一的差別是koa中使用的是yeild next而我們這里用的是yield *next
其實在koa中yeild next和yeild *next效果是等價的,這主要得益于co庫,我們以后再來好好扯扯這小玩意~
總結
以上是生活随笔為你收集整理的generator探幽(1)--koa中间件机制浅析的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: org.hibernate.Mappin
- 下一篇: 10.10SSD安装盘clover 下开