欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

如何在Express4.x中愉快地使用async的方法

 更新時間:2020年11月18日 09:59:49   作者:descire  
這篇文章主要介紹了如何在Express4.x中愉快地使用async的方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

前言

為了能夠更好地處理異步流程,一般開發(fā)者會選擇 async 語法。在 express 框架中可以直接利用 async 來聲明中間件方法,但是對于該中間件的錯誤,無法通過錯誤捕獲中間件來劫持到。

錯誤處理中間件

 const express = require('express');
 const app = express();
 const PORT = process.env.PORT || 3000;

 app.get('/', (req, res) => {
  const message = doSomething();
  res.send(message);
 });

 // 錯誤處理中間件
 app.use(function (err, req, res, next) {
  return res.status(500).send('內(nèi)部錯誤!');
 });

 app.listen(PORT, () => console.log(`app listening on port ${PORT}`));

以上述代碼為例,中間件方法并沒有通過 async 語法來聲明,如果 doSomething 方法內(nèi)部拋出異常,那么就可以在錯誤處理中間件中捕獲到錯誤,從而進(jìn)行相應(yīng)地異常處理。

 app.get('/', async (req, res) => {
  const message = doSomething();
  res.send(message);
 });

而采用 async 語法來聲明中間件時,一旦 doSomething 內(nèi)部拋出異常,則錯誤處理中間件無法捕獲到。

雖然可以利用 process 監(jiān)聽 unhandledRejection 事件來捕獲,但是無法正確地處理后續(xù)流程。

try/catch

對于 async 聲明的函數(shù),可以通過 try/catch 來捕獲其內(nèi)部的錯誤,再使用 next 函數(shù)將錯誤遞交給錯誤處理中間件,即可處理該場景:

 app.get('/', async (req, res, next) => {
  try {
   const message = doSomething();
   res.send(message);
  } catch(err) {
   next(err);
  }
 });

「 這種寫法簡單易懂,但是滿屏的 try/catch 語法,會顯得非常繁瑣且不優(yōu)雅。 」

高階函數(shù)

對于基礎(chǔ)扎實的開發(fā)來說,都知道 async 函數(shù)最終返回一個 Promise 對象,而對于 Promsie 對象應(yīng)該利用其提供的 catch 方法來捕獲異常。

那么在將 async 語法聲明的中間件方法傳入 use 之前,需要包裹一層 Promise 函數(shù)的異常處理邏輯,這時就需要利用高階函數(shù)來完成這樣的操作。

 function asyncUtil(fn) {
  return function asyncUtilWrap(...args) {
   const fnReturn = fn(args);
   const next = args[args.length - 1];
   return Promise.resolve(fnReturn).catch(next);
  }
 }

 app.use(asyncUtil(async (req, res, next) => {
  const message = doSomething();
  res.send(message);
 }));

相比較第一種方法, 「 高階函數(shù)減少了冗余代碼,在一定程度上提高了代碼的可讀性。

上述兩種方案基于扎實的 JavaScript 基礎(chǔ)以及 Express 框架的熟練使用,接下來從源碼的角度思考合適的解決方案。

中間件機(jī)制

Express 中主要包含三種中間件:

  • 應(yīng)用級別中間件
  • 路由級別中間件
  • 錯誤處理中間件
app.use = function use(fn) {
 var path = '/';

 // 省略參數(shù)處理邏輯
 ...

 // 初始化內(nèi)置中間件
 this.lazyrouter();
 var router = this._router;

 fns.forEach(function (fn) {
  // non-express app
  if (!fn || !fn.handle || !fn.set) {
   return router.use(path, fn);
  }

  ...

 }, this);

 return this;
};

應(yīng)用級別中間件通過 app.use 方法注冊, 「 其本質(zhì)上也是調(diào)用路由對象上的中間件注冊方法,只不過其默認(rèn)路由為 '/' 」 。

proto.use = function use(fn) {
 var offset = 0;
 var path = '/';

 // 省略參數(shù)處理邏輯
 ...

 var callbacks = flatten(slice.call(arguments, offset));

 for (var i = 0; i < callbacks.length; i++) {
  var fn = callbacks[i];

  ...

  // add the middleware
  debug('use %o %s', path, fn.name || '<anonymous>')

  var layer = new Layer(path, {
   sensitive: this.caseSensitive,
   strict: false,
   end: false
  }, fn);

  layer.route = undefined;

  this.stack.push(layer);
 }

 return this;
};

中間件的所有注冊方式最終會調(diào)用上述代碼,根據(jù) path 和中間件處理函數(shù)生成 layer 實例,再通過棧來維護(hù)這些 layer 實例。

// 部分核心代碼
proto.handle = function handle(req, res, out) {
 var self = this;
 var idx = 0;
 var stack = self.stack;

 next();

 function next(err) {
  var layerError = err === 'route'
   ? null
   : err;
  
  if (idx >= stack.length) {
   return;
  }

  var path = getPathname(req);

  // find next matching layer
  var layer;
  var match;
  var route;

  while (match !== true && idx < stack.length) {
   layer = stack[idx++];
   match = matchLayer(layer, path);
   route = layer.route;

   if (match !== true) {
    continue;
   }

  }

  // no match
  if (match !== true) {
   return done(layerError);
  }

  // this should be done for the layer
  self.process_params(layer, paramcalled, req, res, function (err) {
   if (err) {
    return next(layerError || err);
   }

   if (route) {
    return layer.handle_request(req, res, next);
   }

   trim_prefix(layer, layerError, layerPath, path);
  });
 }

 function trim_prefix(layer, layerError, layerPath, path) {

  if (layerError) {
   layer.handle_error(layerError, req, res, next);
  } else {
   layer.handle_request(req, res, next);
  }
 }
};

Express 內(nèi)部通過 handle 方法來處理中間件執(zhí)行邏輯,其利用 「 閉包的特性 」 緩存 idx 來記錄當(dāng)前遍歷的狀態(tài)。

該方法內(nèi)部又實現(xiàn)了 next 方法來匹配當(dāng)前需要執(zhí)行的中間件,從遍歷的代碼可以明白 「 中間件注冊的順序是非常重要的 」 。

如果該流程存在異常,則調(diào)用 layer 實例的 handle.error 方法,這里仍然是 「 遵循了 Node.js 錯誤優(yōu)先的設(shè)計理念 」 :

Layer.prototype.handle_error = function handle_error(error, req, res, next) {
 var fn = this.handle;

 if (fn.length !== 4) {
  // not a standard error handler
  return next(error);
 }

 try {
  fn(error, req, res, next);
 } catch (err) {
  next(err);
 }
};

內(nèi)部通過判斷函數(shù)的形參個數(shù)過濾掉非錯誤處理中間件」。
如果 next 函數(shù)內(nèi)部沒有異常情況,則調(diào)用 layer 實例的 handle_request 方法:

Layer.prototype.handle_request = function handle(req, res, next) {
 var fn = this.handle;

 if (fn.length > 3) {
  // not a standard request handler
  return next();
 }

 try {
  fn(req, res, next);
 } catch (err) {
  next(err);
 }
};

 「 handle 方法初始化執(zhí)行了一次 next 方法,但是該方法每次調(diào)用最多只能匹配一個中間件 」 ,所以在執(zhí)行 handle_error 和 handle_request 方法時,會將 next 方法透傳給中間件,這樣開發(fā)者就可以通過手動調(diào)用 next 方法的方式來執(zhí)行接下來的中間件。

從上述中間件的執(zhí)行流程中可以知曉, 「 用戶注冊的中間件方法在執(zhí)行的時候都會包裹一層 try/catch,但是 try/catch 無法捕獲 async 函數(shù)內(nèi)部的異常,這也就是為什么 Express 中無法通過注冊錯誤處理中間件來攔截到 async 語法聲明的中間件的異常的原因 」 。

修改源碼

找到本質(zhì)原因之后,可以通過修改源碼的方法來進(jìn)行適配:

Layer.prototype.handle_request = function handle(req, res, next) {
 var fn = this.handle;

 if (fn.length > 3) {
  // not a standard request handler
  return next();
 }
 // 針對 async 語法函數(shù)特殊處理
 if (Object.prototype.toString.call(fn) === '[object AsyncFunction]') {
  return fn(req, res, next).catch(next);
 }

 try {
  fn(req, res, next);
 } catch (err) {
  next(err);
 }
};

上述代碼在 handle_request 方法內(nèi)部判斷了中間件方法通過 async 語法聲明的情況,從而采用 Promise 對象的 catch 方法來向下傳遞異常。

這種方式可以減少上層冗余的代碼,但是實現(xiàn)該方式,可能需要 fork 一份 Express4.x 的源碼,然后發(fā)布一個修改之后的版本,后續(xù)還要跟進(jìn)官方版本的新特性,相應(yīng)的維護(hù)成本非常高。

express5.x 中將 router 部分剝離出了單獨的路由庫 -- router

AOP(面向切面編程)

為了解決上述方案存在的問題,我們可以嘗試?yán)?AOP 技術(shù)在不修改源碼的基礎(chǔ)上對已有方法進(jìn)行增強(qiáng)。

app.use(async function () {
 const message = doSomething();
 res.send(message);
})

以注冊應(yīng)用級別中間件為例,可以對 app.use 方法進(jìn)行 AOP 增強(qiáng):

const originAppUseMethod = app.use.bind(app);
app.use = function (fn) {
 if (Object.prototype.toString.call(fn) === '[object AsyncFunction]') {
  const asyncWrapper = function(req, res, next) {
   fn(req, res, next).then(next).catch(next);
  }
  return originAppUseMethod(asyncWrapper);
 }
 return originAppUseMethod(fn);
}

前面源碼分析的過程中,app.use 內(nèi)部是有 this 調(diào)用的,所以這里需要 「 利用 bind 方法來避免后續(xù)調(diào)用過程中 this 指向出現(xiàn)問題。

然后就是利用 AOP 的核心思想,重寫原始的 app.use 方法,通過不同的分支邏輯代理到原始的 app.use 方法上。

該方法相比較修改源碼的方式,維護(hù)成本低。但是缺點也很明顯,需要重寫所有可以注冊中間件的方法,不能夠像修改源碼那樣一步到位。

寫在最后

本文介紹了 Express 中使用 async 語法的四種解決方案:

  • try/catch
  • 高階函數(shù)
  • 修改源碼
  • AOP

除了 try/catch 方法性價比比較低,其它三種方法都需要根據(jù)實際情況去取舍,舉個栗子:

如果你需要寫一個 Express 中間件提供給各個團(tuán)隊使用,那么修改源碼的方式肯定走不通,而 AOP 的方式對于你的風(fēng)險太大,相比較下,第二種方案是最佳的實踐方案。

到此這篇關(guān)于如何在Express4.x中愉快地使用async的方法的文章就介紹到這了,更多相關(guān)Express4.x使用async內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:

相關(guān)文章

  • Nodejs 數(shù)組的隊列以及forEach的應(yīng)用詳解

    Nodejs 數(shù)組的隊列以及forEach的應(yīng)用詳解

    這篇文章主要介紹了Nodejs 數(shù)組的隊列以及forEach的應(yīng)用詳解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-02-02
  • node.js中module模塊的功能理解與用法實例分析

    node.js中module模塊的功能理解與用法實例分析

    這篇文章主要介紹了node.js中module模塊的功能理解與用法,結(jié)合實例形式分析了node.js module模塊的基本功能、原理、用法及相關(guān)操作注意事項,需要的朋友可以參考下
    2020-02-02
  • node.JS md5加密中文與php結(jié)果不一致的解決方法

    node.JS md5加密中文與php結(jié)果不一致的解決方法

    本篇文章主要介紹了node.JS md5加密中文與php結(jié)果不一致的解決方法,具有很好的參考價值。下面跟著小編一起來看下吧
    2017-05-05
  • 創(chuàng)建簡單的node服務(wù)器實例(分享)

    創(chuàng)建簡單的node服務(wù)器實例(分享)

    下面小編就為大家?guī)硪黄獎?chuàng)建簡單的node服務(wù)器實例(分享)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-06-06
  • node中使用shell腳本的方法步驟

    node中使用shell腳本的方法步驟

    這篇文章主要介紹了node中使用shell腳本的方法步驟,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-03-03
  • Node.js 異步編程之 Callback介紹(一)

    Node.js 異步編程之 Callback介紹(一)

    這篇文章主要介紹了Node.js 異步編程之 Callback介紹(一),本文用實例講解Callback的相關(guān)知識,本文是第一篇,下一篇小編會跟進(jìn),需要的朋友可以參考下
    2015-03-03
  • 基于Koa(nodejs框架)對json文件進(jìn)行增刪改查的示例代碼

    基于Koa(nodejs框架)對json文件進(jìn)行增刪改查的示例代碼

    這篇文章主要介紹了基于Koa(nodejs框架)對json文件進(jìn)行增刪改查的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-02-02
  • node.js實現(xiàn)微信JS-API封裝接口的示例代碼

    node.js實現(xiàn)微信JS-API封裝接口的示例代碼

    這篇文章主要介紹了node.js實現(xiàn)微信JS-API封裝接口的示例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-09-09
  • 詳解nodejs 開發(fā)企業(yè)微信第三方應(yīng)用入門教程

    詳解nodejs 開發(fā)企業(yè)微信第三方應(yīng)用入門教程

    這篇文章主要介紹了詳解nodejs 開發(fā)企業(yè)微信第三方應(yīng)用入門教程,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2019-03-03
  • 淺談Nodejs中的作用域問題

    淺談Nodejs中的作用域問題

    在JS中有全局作用域和函數(shù)作用域,而在Nodejs中也自己的作用域,分為全局作用域(global)和模塊作用域。本文將對Nodejs中的作用域進(jìn)行介紹,需要的朋友一起來看下吧
    2016-12-12

最新評論