這次帶給大家nodejs操作express中的next,nodejs操作express中next的注意事項有哪些,下面就是實戰案例,一起來看一下。
相信很多人對於nodejs對express中next不是很了解,大多數人不知道其中的next有什麼作用,在什麼情況下使用,今天文章中就給大家詳細介紹下其使用方法,感興趣的一起來了解下。
最近公司在使用node做前後端分離,採用的web框架是express,所以對express框架進行了深入的了解,前段時間寫了篇關於express路由的文章,但是在那篇文章中貌似少了一個很重要的內容,就是express的next,所以今天單獨來談談express的next。
關於next主要從三點來進行說明:
# next的作用是什麼?
我們應該在何時使用next?
- next的內部實作機制是什麼?
Next的作用
我們在定義express中間件函數的時候都會將第三個參數定義為next,這個next就是我們今天的主角,next函數主要負責將控制權交給下一個中間件,如果當前中間件沒有終結請求,且next沒有被調用,那麼請求將被掛起,後邊定義的中間件將不會得到被執行的機會。
何時使用Next
從上邊的描述我們已經知道,next函數主要是用來確保所有註冊的中間件被一個接一個的執行,那麼我們就應該在所有的中間件中調用next函數,但有一個特例,如果我們定義的中間件終結了本次請求,那就不應該再呼叫next函數,否則就可能會出問題,我們來看段碼
app.get('/a', function(req, res, next) { res.send('sucess'); next(); }); // catch 404 and forward to error handler app.use(function(req, res, next) { console.log(404); var err = new Error('Not Found'); err.status = 404; next(err); }); app.use(function(err, req, res, next) { res.status(err.status || 500); res.render('error', { message: err.message, error: {} }); });
發送請求"/a",控制台列印日誌如下:
404 GET /a 500 6.837 ms - - Error: Can't set headers after they are sent. at ServerResponse.OutgoingMessage.setHeader (_http_outgoing.js:345:11)
為什麼程式碼會拋異常呢,就是因為我們在res.send之後呼叫了next函數,雖然我們本次的請求已經被終止,但後邊的404中間件依舊會被執行,而後邊的中間件試圖去向res的headers中加入屬性值,所以就會拋出上邊的異常。
讀到這裡你可能會有個疑問,如果我不在res.send後邊呼叫next函數,那後邊定義的404中間件是不是永遠都不會被執行到。現在我們刪除res.send後邊next函數調用,發送請求"/xxx",我們會發現404中間件被執行了,(ㄒoㄒ),這不是和我們之前說的矛盾了嗎,我們的自定義中間件沒有呼叫next,但後邊定義的中間件仍舊被執行了,這究竟是為什麼呢。看來只能求助原始碼了~~~
Next的內部機制
function next(err) { ... //此处源码省略 // find next matching layer var layer; var match; var route; while (match !== true && idx <p style="text-align: left;"> 上邊就是express中next的源碼,為了更容易說明問題,程式碼進行了刪減。從上邊的源碼可以發現,next函數內部有個<a href="http://www.php.cn/code/9671.html" target="_blank">while循環</a>,每次循環都會從stack中拿出一個layer,這個layer中包含了路由和中間件信息,然後就會用layer和請求的path就行匹配,如果匹配成功就會執行layer.handle_request,呼叫中間件函數。但如果匹配失敗,就會循環下一個layer(即中間件)。 </p><p style="text-align: left;"> 現在我們就能解釋上邊提出的問題了,為什麼我們的自訂中間件中沒呼叫next函數,但後邊的404中間件仍舊會被執行到,因為我們請求的"/xxx"匹配不到我們註冊的"/a"路由中間件,所以while循環會繼續往下執行,匹配404中間件成功,所以會執行404中間件。 </p><p style="text-align: left;"> 注意:app.use註冊的中間件,如果path參數為空,則預設為"/",而path為"/"的中間件則預設符合所有的請求。 </p><p style="text-align: left;"> 有一點需要特別指出,其實我們在<a href="http://www.php.cn/code/8256.html" target="_blank">定義路由</a>中間件的時候函數的第三個參數next和我們定義非路由中間件的函數的第三個參數next不是同一個next,我們在上邊看到的是非路由中間件的next,而路由中間件的next函數是這樣的</p><pre class="brush:php;toolbar:false">function next(err) { if (err && err === 'route') { return done(); } var layer = stack[idx++]; if (!layer) { return done(err); } if (layer.method && layer.method !== method) { return next(err); } if (err) { layer.handle_error(err, req, res, next); } else { layer.handle_request(req, res, next); } }
这个next比上边的那个next要简单很多,它负责同一个路由的多个中间件的控制权的传递,并且它会接收一个参数"route",如果调用next(“route”),则会跳过当前路由的其它中间件,直接将控制权交给下一个路由。
最后有必要再说一说next(err),next(err)是如何将控制权传递到错误处理中间件的,从前边的代码我们知道,当调用next(err)是,express内部会调用layer.handle_error,那我们来看看它的源码
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); } };
代码中的fn就是中间件函数,express会对fn的参数个数进行判断,如果参数个数不等于4则认为不是错误处理中间件,则继续调用next(err),这样就会进入到下一个中间件函数,继续进行参数个数判断,如此方式一直到某个中间件函数的参数个数是4,就认为找到了错误处理中间件,然后执行此中间件函数。
相信看了本文案例你已经掌握了方法,更多精彩请关注php中文网其它相关文章!
推荐阅读:
以上是nodejs操作express中的next的詳細內容。更多資訊請關注PHP中文網其他相關文章!

去掉重复并排序的方法:1、使用“Array.from(new Set(arr))”或者“[…new Set(arr)]”语句,去掉数组中的重复元素,返回去重后的新数组;2、利用sort()对去重数组进行排序,语法“去重数组.sort()”。

本篇文章给大家带来了关于JavaScript的相关知识,其中主要介绍了关于Symbol类型、隐藏属性及全局注册表的相关问题,包括了Symbol类型的描述、Symbol不会隐式转字符串等问题,下面一起来看一下,希望对大家有帮助。

怎么制作文字轮播与图片轮播?大家第一想到的是不是利用js,其实利用纯CSS也能实现文字轮播与图片轮播,下面来看看实现方法,希望对大家有所帮助!

本篇文章给大家带来了关于JavaScript的相关知识,其中主要介绍了关于对象的构造函数和new操作符,构造函数是所有对象的成员方法中,最早被调用的那个,下面一起来看一下吧,希望对大家有帮助。

本篇文章给大家带来了关于JavaScript的相关知识,其中主要介绍了关于面向对象的相关问题,包括了属性描述符、数据描述符、存取描述符等等内容,下面一起来看一下,希望对大家有帮助。

方法:1、利用“点击元素对象.unbind("click");”方法,该方法可以移除被选元素的事件处理程序;2、利用“点击元素对象.off("click");”方法,该方法可以移除通过on()方法添加的事件处理程序。

foreach不是es6的方法。foreach是es3中一个遍历数组的方法,可以调用数组的每个元素,并将元素传给回调函数进行处理,语法“array.forEach(function(当前元素,索引,数组){...})”;该方法不处理空数组。

本篇文章给大家带来了关于JavaScript的相关知识,其中主要介绍了关于BOM操作的相关问题,包括了window对象的常见事件、JavaScript执行机制等等相关内容,下面一起来看一下,希望对大家有帮助。


熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

AI Hentai Generator
免費產生 AI 無盡。

熱門文章

熱工具

SublimeText3 英文版
推薦:為Win版本,支援程式碼提示!

MantisBT
Mantis是一個易於部署的基於Web的缺陷追蹤工具,用於幫助產品缺陷追蹤。它需要PHP、MySQL和一個Web伺服器。請查看我們的演示和託管服務。

mPDF
mPDF是一個PHP庫,可以從UTF-8編碼的HTML產生PDF檔案。原作者Ian Back編寫mPDF以從他的網站上「即時」輸出PDF文件,並處理不同的語言。與原始腳本如HTML2FPDF相比,它的速度較慢,並且在使用Unicode字體時產生的檔案較大,但支援CSS樣式等,並進行了大量增強。支援幾乎所有語言,包括RTL(阿拉伯語和希伯來語)和CJK(中日韓)。支援嵌套的區塊級元素(如P、DIV),

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

SublimeText3 Linux新版
SublimeText3 Linux最新版