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

淺談Express異步進化史

 更新時間:2017年09月09日 14:18:22   作者:幻天芒  
本篇文章主要介紹了淺談Express異步進化史 ,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧

1、導言

在 Javascript 的世界里,異步(由于JavaScript的單線程運行,所以JavaScript中的異步是可以阻塞的)無處不在。

Express 是 node 環(huán)境中非常流行的Web服務端框架,有很大比例的 Node Web應用 采用了 Express。

當使用 JavaScript 編寫服務端代碼時,我們無可避免的會大量使用到異步。隨著 JavaScript、Node 的進化,我們的異步處理方式,也就隨之進化。

接下來,我們就來看看 Express 中異步處理的進化過程。

2、JavaScript的異步處理

在異步的世界里,我們需要想辦法獲取的異步方法完畢的通知,那在 JavaScript 中,會有哪些方式呢?

2.1、回調(diào)

回調(diào)是 JS 中最原始,也是最古老的異步通知機制。

function asyncFn(callback) {
 // 利用setTimeout模擬異步
 setTimeout(function () {
  console.log('執(zhí)行完畢');
  callback(); // 發(fā)通知
 }, 2000);
}

asyncFn(function () {
 console.log('我會在2s后輸出');
});

2.2、事件監(jiān)聽

要獲取結(jié)果的函數(shù),監(jiān)聽某個時間。在異步方法完成后,觸發(fā)該事件,達到通知的效果。

2.3、發(fā)布/訂閱

通過觀察者模式,在異步完成時,修改發(fā)布者。這個時候,發(fā)布者會把變更通知到訂閱者。

2.4、Promise

Promise 是回調(diào)函數(shù)的改進。使用它, 我們可以將異步平行化,避免回調(diào)地獄。

function asyncFn() {
 return new Promise((resolve, reject) => {
  // 利用setTimeout模擬異步
  setTimeout(function () {
   console.log('執(zhí)行完畢');
   resolve(); // 發(fā)通知(是否有感覺到回調(diào)的影子?)
  }, 2000);
 });
}

asyncFn()
 .then(function () {
  console.log('我會在2s后輸出');
 });

2.5、生成器(Generator)

Generator 函數(shù)是 ES6 提供的一種異步編程解決方案。

以下代碼只是簡單演示,實際上 Generator 的使用過程,相對是比較復雜的,這是另外一個話題,本文暫且不表。

function asyncFn() {
 return new Promise((resolve, reject) => {
  // 利用setTimeout模擬異步
  setTimeout(function () {
   console.log('執(zhí)行完畢');
   resolve(); // 發(fā)通知(是否有感覺到回調(diào)的影子?)
  }, 2000);
 });
}

function* generatorSync() {
 var result = yield asyncFn();
}

var g = generatorSync();
g.next().value.then(()=>{
 console.log('我會在2s后輸出');
});

2.6、async...await

可以說是當前 JavaScript 中,處理異步的最佳方案。

function asyncFn() {
 return new Promise((resolve, reject) => {
  // 利用setTimeout模擬異步
  setTimeout(function () {
   console.log('執(zhí)行完畢');
   resolve(); // 發(fā)通知(是否有感覺到回調(diào)的影子?)
  }, 2000);
 });
}

async function run(){
 await asyncFn();
 console.log('我會在2s后輸出');
}

run();

3、Express中的異步處理

在Express中,我們一般常用的是方案是:回調(diào)函數(shù)、Promise、以及async...await。

為了搭建演示環(huán)境,通過 express-generator 初始化一個express項目。一般的服務端項目,都是路由調(diào)用業(yè)務邏輯。所以,我們也遵循這個原則:

打開 routs/index.js,我們會看到如下內(nèi)容,以下Demo就以此文件來做演示。

var express = require('express');
var router = express.Router();

/* GET home page. */
router.get('/', function(req, res, next) {
 res.render('index', { title: 'Express' });
});

module.exports = router;

3.1、回調(diào)函數(shù)處理Express異步邏輯

在 Express 中,路由可以加載多個中間件,所以我們可以把業(yè)務邏輯按照中間件的寫法進行編寫。這樣通過一層層的next,就能非常方便的拆分異步邏輯。

var express = require('express');
var router = express.Router();

function asyncFn(req, res, next) {
 setTimeout(() => {
  req.user = {}; // 設置當前請求的用戶
  next();
 }, 2000);
}

function asyncFn2(req, res, next) {
 setTimeout(() => {
  req.auth = {}; // 設置用戶權(quán)限
  next();
 }, 2000);
}

function asyncFn3(req, res, next) {
 setTimeout(() => {
  res.locals = { title: 'Express Async Test' }; // 設置數(shù)據(jù)
  res.render('index'); // 響應
 }, 2000);
}

/* GET home page. */
router.get('/', asyncFn, asyncFn2, asyncFn3); // 一步步執(zhí)行中間件

module.exports = router;

3.2、Promise 處理Express異步邏輯

該方案中,將多個業(yè)務邏輯,包裝為返回 Promise 的函數(shù)。通過業(yè)務方法進行組合調(diào)用,以達到一進一出的效果。

var express = require('express');
var router = express.Router();

function asyncFn(req, res) {
 return new Promise((resolve, reject) => {
  setTimeout(() => {
   req.user = {}; // 設置當前請求的用戶
   resolve(req);
  }, 2000);
 });
}

function asyncFn2(req) {
 return new Promise((resolve, reject) => {
  setTimeout(() => {
   req.auth = {}; // 設置用戶權(quán)限
   resolve();
  }, 2000);
 });
}

function asyncFn3(res) {
 return new Promise((resolve, reject) => {
  setTimeout(() => {
   res.locals = { title: 'Express Async Test' }; // 設置數(shù)據(jù)
   res.render('index'); // 響應
  }, 2000);
 });
}

function doBizAsync(req, res, next) {
 asyncFn(req)
  .then(() => asyncFn2(req))
  .then(() => asyncFn3(res))
  .catch(next); // 統(tǒng)一異常處理
};

/* GET home page. */
router.get('/', doBizAsync);

module.exports = router;

3.3、async...await 處理Express異步邏輯

實際上,該方案也是需要 Promise 的支持,只是寫法上,更直觀,錯誤處理也更直接。

需要注意的是,Express是早期的方案,沒有對async...await進行全局錯誤處理,所以可以采用包裝方式,進行處理。

var express = require('express');
var router = express.Router();

function asyncFn(req) {
 return new Promise((resolve, reject) => {
  setTimeout(() => {
   req.user = {}; // 設置當前請求的用戶
   resolve(req);
  }, 2000);
 });
}

function asyncFn2(req) {
 return new Promise((resolve, reject) => {
  setTimeout(() => {
   req.auth = {}; // 設置用戶權(quán)限
   resolve();
  }, 2000);
 });
}

function asyncFn3(res) {
 return new Promise((resolve, reject) => {
  setTimeout(() => {

  }, 2000);
 });
}

async function doBizAsync(req, res, next) {
 var result = await asyncFn(req);
 var result2 = await asyncFn2(req);
 res.locals = { title: 'Express Async Test' }; // 設置數(shù)據(jù)
 res.render('index'); // 響應
};

const tools = {
 asyncWrap(fn) {
  return (req, res, next) => {
   fn(req, res, next).catch(next); // async...await在Express中的錯誤處理
  }
 }
};

/* GET home page. */
router.get('/', tools.asyncWrap(doBizAsync)); // 需要用工具方法包裹一下

module.exports = router;

4、總結(jié)

雖然 koa 對更新、更好的用法(koa是generator,koa2原生async)支持的更好。但作為從 node 0.x 開始跟的我,對 Express 還是有特殊的好感。

以上的一些方案,已經(jīng)與 koa 中使用無異,配合 Express 龐大的生態(tài)圈,無異于如虎添翼。

本文Github地址

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Nodejs中的this詳解

    Nodejs中的this詳解

    本文給大家匯總介紹了下nodejs中的this,nodejs中的this和在瀏覽器中javascript中的this還是有很大的差異的,小伙伴們可以研究下
    2016-03-03
  • node.js中的http.request.end方法使用說明

    node.js中的http.request.end方法使用說明

    這篇文章主要介紹了node.js中的http.request.end方法使用說明,本文介紹了http.request.end的方法說明、語法、接收參數(shù)、使用實例和實現(xiàn)源碼,需要的朋友可以參考下
    2014-12-12
  • windows系統(tǒng)下安裝yarn的詳細教程

    windows系統(tǒng)下安裝yarn的詳細教程

    yarn是一個新的JS包管理工具,它的出現(xiàn)是為了彌補npm的一些缺陷,下面這篇文章主要給大家介紹了關(guān)于windows系統(tǒng)下安裝yarn的詳細教程,文中通過圖文介紹的非常詳細,需要的朋友可以參考下
    2023-02-02
  • nodejs調(diào)取微信收貨地址的方法

    nodejs調(diào)取微信收貨地址的方法

    這篇文章主要為大家詳細介紹了nodejs調(diào)取微信收貨地址的方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-12-12
  • npm install --save 、--save-dev 、-D、-S 的區(qū)別與NODE_ENV的配置方法

    npm install --save 、--save-dev 、-D、-S&nb

    這篇文章主要介紹了npm install --save 、--save-dev 、-D、-S 的區(qū)別與NODE_ENV的配置方法,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-08-08
  • 使用puppeteer破解極驗的滑動驗證碼

    使用puppeteer破解極驗的滑動驗證碼

    這篇文章主要介紹了利用puppeteer破解極驗的滑動驗證功能,基本流程代碼實現(xiàn)給大家介紹的非常詳細,需要的朋友可以參考下
    2018-02-02
  • 從零開始學習Node.js系列教程之設置HTTP頭的方法示例

    從零開始學習Node.js系列教程之設置HTTP頭的方法示例

    這篇文章主要介紹了Node.js設置HTTP頭的方法,詳細分析了常見HTTP頭的功能、原理及相關(guān)設置操作技巧,需要的朋友可以參考下
    2017-04-04
  • 如何用Node.js編寫內(nèi)存效率高的應用程序

    如何用Node.js編寫內(nèi)存效率高的應用程序

    這篇文章主要介紹了如何用Node.js編寫內(nèi)存效率高的應用程序,對Node.js感興趣的同學,可以參考下
    2021-04-04
  • 十大 Node.js 的 Web 框架(快速提升工作效率)

    十大 Node.js 的 Web 框架(快速提升工作效率)

    Node.js 系統(tǒng)含有多種不同的結(jié)構(gòu),如 MVC、全棧、REST API 和生成器等。這些結(jié)構(gòu)不僅提升了 Web 應用的開發(fā)效率,也優(yōu)化了開發(fā)過程。在這里,我們收集整理了十個高效的 Node.js 框架,希望對你有幫助。
    2017-06-06
  • 用Node寫一條配置環(huán)境的指令

    用Node寫一條配置環(huán)境的指令

    這篇文章主要介紹了用Node寫一條配置環(huán)境的指令,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-11-11

最新評論