鍍金池/ 問(wèn)答/Python  Linux  HTML/ koa2 框架中的中間件同步還是異步的問(wèn)題?

koa2 框架中的中間件同步還是異步的問(wèn)題?

問(wèn)題 1:為什么 koa2 框架中的中間件要用 async 的形式寫,很少見用同步模式(即不加 async)? 如:

app.use(async (ctx, next) => {
  const start = new Date()
  await next()
  const ms = new Date() - start
  console.log(`${ctx.method} ${ctx.url} - ${ms}ms`)
})

查閱了一些資料,看到阮一峰的 koa 教程中,有同步寫法的例子:

const one = (ctx, next) => {
  console.log('>> one');
  next();
  console.log('<< one');
}

const two = (ctx, next) => {
  console.log('>> two');
  next(); 
  console.log('<< two');
}

const three = (ctx, next) => {
  console.log('>> three');
  next();
  console.log('<< three');
}

app.use(one);
app.use(two);
app.use(three);

/*
output:
>> one
>> two
>> three
<< three
<< two
<< one
*/

這里完全沒(méi)有 async 的出現(xiàn),阮神只是在后面提到: “迄今為止,所有例子的中間件都是同步的,不包含異步操作。如果有異步操作(比如讀取數(shù)據(jù)庫(kù)),中間件就必須寫成 async 函數(shù)?!?/p>

接著找資料發(fā)現(xiàn) koa2 中間件的 next 返回的是 promise,因?yàn)樵创a中的 dispatch(i)返回的就是 promise, 這樣我又有一個(gè)疑問(wèn),
問(wèn)題 2:為什么要設(shè)計(jì)成返回 promise ?不返回 promise 就達(dá)不到中間件串聯(lián)的效果嗎? 因?yàn)橹虚g件的執(zhí)行在理解上是一個(gè)同步的過(guò)程,所以設(shè)計(jì)成異步要怎么去理解??

問(wèn)題 3:是不是用 koa2 寫的代碼都存在很多 async 的 function(公司項(xiàng)目代碼到處都是 async 的,返回輸出 json 的時(shí)候也需要 async 嗎?)? 有沒(méi)有不需要寫 async 的場(chǎng)景或者例子?

回答
編輯回答
過(guò)客
  1. 異步函數(shù)也是函數(shù),只是會(huì)返回一個(gè)promise對(duì)象,里面可能存在promise串聯(lián)的情況
  2. promise最適合做異步操作,讓異步操作看起來(lái)像是同步的,不然通過(guò)回掉會(huì)很麻煩,邏輯也不夠清晰
  3. 你可以完全忽略同步函數(shù)和異步函數(shù),都可以接收的,返回值是promise就等到promise完成
2018年4月18日 08:19
編輯回答
不將就

1.因?yàn)殡x職1中要await next(),作為一個(gè)中間件,你很難確定你的next里面到底是同步方法還是異步方法,如果你需要等待其執(zhí)行完成,那最好是當(dāng)異步方法處理,當(dāng)然你也可以不用async/await。
2.返回Promise是目前處理異步最適合的方式,沒(méi)有理解中間件的執(zhí)行是一個(gè)同步的過(guò)程這句話,執(zhí)行script是一個(gè)同步的過(guò)程,但里面很可能包含異步操作。
3.async function并不意味著其內(nèi)部就必須含有異步方法,輸出json并不需要async,更多的可能是因?yàn)榉椒▋?nèi)部可能出現(xiàn)異步方法并希望使用await等待其執(zhí)行完成,另外async function返回的是一個(gè)Promise

2017年10月17日 06:08
編輯回答
過(guò)客

首先,可以先去看一下generator,async,await 相關(guān)的基礎(chǔ)知識(shí)和原理,看看阮一峰的教程就行。async 主要是解決了異步操作的問(wèn)題,我們可以不寫Promise和callback,可以像調(diào)用同步方法一樣執(zhí)行一個(gè)異步方法。接下來(lái)回答問(wèn)題
1、當(dāng)然可以不用async和await,即便是你的中間件里有異步操作,也可以通過(guò)promise來(lái)處理,只不過(guò) async 的語(yǔ)法寫起來(lái)更像同步的,看起來(lái)更清爽
2、因?yàn)?await 后必須是一個(gè) promise,所以next才設(shè)計(jì)成了promise(所以還是建議你先去看看原理)。不使用promise也可以達(dá)到串聯(lián)效果,那么就要利用回調(diào)了;中間件的執(zhí)行你要考慮一種情況,在a中間件里,我需要調(diào)用一個(gè)接口,接口返回后再進(jìn)入下一個(gè)中間件,這其實(shí)就是一個(gè)異步的情況
3、還是回到最開始,async 只是一個(gè)方案,讓我們寫異步方法更爽,你可以不用

2017年7月10日 14:23