Nodejs libuv運(yùn)行原理詳解
前言
這應(yīng)該是Nodejs的運(yùn)行原理的第7篇分享,這篇過(guò)后,短時(shí)間內(nèi)不會(huì)再分享Nodejs的運(yùn)行原理,會(huì)停更一段時(shí)間,PS:不是不更,而是會(huì)開挖新的坑,最近有在研究RPG Maker MV,區(qū)塊鏈,云計(jì)算,可能會(huì)更新一些相關(guān)文章,或者相關(guān)教學(xué)。
回到正題,異步編程的難點(diǎn)在于請(qǐng)求與響應(yīng)不是按順序發(fā)生的。以http server 為例,異步編程賦予了server 高并發(fā)的品質(zhì),而且他可以以很小的資源代價(jià),不斷地接受和處理請(qǐng)求。但是快速處理請(qǐng)求不表示快速地返回請(qǐng)求=>高并發(fā)不等同于快速反饋。
在Nodejs中,libuv則為異步編程的實(shí)現(xiàn)提供了可能。libuv為builtin modules 提供了API,這些API用來(lái)支撐請(qǐng)求和數(shù)據(jù)的返回的異步處理方式。
這一篇分享,我們主要討論libuv的運(yùn)行原理,從兩個(gè)角度出發(fā):
1) libuv的架構(gòu)
2) 案例,從細(xì)節(jié)的角度看libuv是如何對(duì)待不同I/O請(qǐng)求,按照不同的方式來(lái)完成異步請(qǐng)求和數(shù)據(jù)返回的。
Libuv的架構(gòu)
從左往右可分為兩部分,Network I/O的相關(guān)請(qǐng)求,另一部分File I/O,DNS Ops和User Code組成。
上圖展示了libuv細(xì)節(jié)的流程,圖中代碼很簡(jiǎn)單,包括2個(gè)部分:
1. server.listen()是用來(lái)創(chuàng)建TCP server時(shí),通常放在最后一步執(zhí)行的代碼。主要指定服務(wù)器工作的端口以及回調(diào)函數(shù)。
2. fs.open()是用異步的方式打開一個(gè)文件。
選擇兩個(gè)示例很簡(jiǎn)單,因?yàn)閘ibuv架構(gòu)圖可視:libuv對(duì) Network I/O和 File I/O采用不同的機(jī)制。
上圖右半部分,主要分成兩個(gè)部分:
1. 主線程:主線程也是node啟動(dòng)時(shí)執(zhí)行的現(xiàn)成。node啟動(dòng)時(shí),會(huì)完成一系列的初始化動(dòng)作,啟動(dòng)V8 engine,進(jìn)入下一個(gè)循環(huán)。
2. 線程池:線程池的數(shù)量可以通過(guò)環(huán)境變量UV_THREADPOOL_SIZE配置,最大不超過(guò)128個(gè),默認(rèn)為4個(gè)。
Network I/O
V8 engine執(zhí)行從server.listen() 開始,調(diào)用builtin module Tcp_wrap 的過(guò)程。
在創(chuàng)建TCP鏈接的過(guò)程中,libuv直接參與Tcp_wrap.cc函數(shù)中的 TCPWrap::listen() 調(diào)用uv_listen()開始到執(zhí)行uv_io_start()結(jié)束。看起來(lái)很短暫的過(guò)程,其實(shí)是類似linux kernel的中斷處理機(jī)制。
uv_io_start()負(fù)載將handle插入到處理的water queue中。這樣的好處是請(qǐng)求能夠立即得到處理。中斷處理機(jī)制里面的下半部分與數(shù)據(jù)處理操作相似,交由主線程去完成處理。
代碼邏輯很簡(jiǎn)單,查看loop中是否包含handle,如果有遍歷default loop。
File I/O
這里我們研究一下 File I/O。
同Network I/O一樣,我們的應(yīng)用所依賴的fs模塊,后面有一個(gè)builtin module Node_file.cc作為支撐。 Node_file.cc包含了各種我們常用的文件操作的接口,例如open, read, write, chmod,chown等。但同時(shí),它們都支持異步模式。 我們通過(guò)Node_file.cc中的Open()函數(shù)來(lái)研究一下具體的實(shí)現(xiàn)細(xì)節(jié)。
如果你用類似source insight之類的代碼閱讀工具跟蹤一下代碼調(diào)用順序,會(huì)很容易發(fā)現(xiàn)對(duì)于異步模式,Open()函數(shù)會(huì)在一系列輔助操作之后,進(jìn)入函數(shù)uv_fs_open(),并且傳入了一個(gè)FSReqWrap的對(duì)象。
FSReqWrap(),從名字可以看得出來(lái),這是一個(gè)wrap,且是與FS相關(guān)的請(qǐng)求。也就是說(shuō),它基于某一個(gè)現(xiàn)成的機(jī)制來(lái)實(shí)現(xiàn)與FS相關(guān)的請(qǐng)求操作。這個(gè)現(xiàn)成的機(jī)制就是ReqWrap。好吧,它也是個(gè)wrap。乘你還沒瘋的時(shí)候,看一下圖6吧。這里完整展示了FSReqWrap類繼承關(guān)系。
除了FSReqWrap,還有其它Wrap,例如PipeConnectWrap,TCPConnectWrap等等。每個(gè)Wrap均為一種請(qǐng)求類型服務(wù)。 但是這些wrap,都是node自身的行為,而與libuv相關(guān)的是什么呢?上圖中表示出了FSReqWrap關(guān)鍵的數(shù)據(jù)結(jié)構(gòu) uv_fs_s req__。
讓我們把目光回到uv_fs_open()。在調(diào)用這個(gè)函數(shù)時(shí), req__作為其一個(gè)重要的參數(shù)被傳遞進(jìn)去。而在uv_fs_open()內(nèi)部,req__則被添加到work queue的末尾中去。圖3 thread pool中的thread會(huì)去領(lǐng)取這些request進(jìn)行處理。 每個(gè)request很像一個(gè)粘貼板,它將event loop, work queue,每個(gè)請(qǐng)求的處理函數(shù)(work()),以及請(qǐng)求結(jié)束處理函數(shù)(done())綁定在一起。綁定的操作在uv__work_submit()中完成。 例如對(duì)于這里的req__,綁定在它身上的work()為uv__fs_work(), done()為uv__fs_done()。
這里有一個(gè)比較有意思的問(wèn)題值得額外看一下。我們的thread pool是在什么時(shí)候建立的呢?
答案是:在第一次異步調(diào)用uv__work_submit()時(shí)。
每個(gè)thead的入口函數(shù)是 Threadpool.c中的worker()。工作邏輯比較簡(jiǎn)單,依次取出work queue中的請(qǐng)求,執(zhí)行綁定在該請(qǐng)求上的work()函數(shù)。 前面我們提到的綁定在請(qǐng)求上的done()函數(shù)在哪里執(zhí)行呢?這也是一個(gè)比較有意思的操作。libuv通過(guò)uv_async_send()通知event loop去執(zhí)行相應(yīng)的callback函數(shù),也即我們綁定在request上的done()函數(shù)。uv__work_done()用于完成這樣的操作。
uv_async_send()與主線程之間通過(guò)PIPE通信。
我在這一小節(jié)以一個(gè)FSReqWrap以及Open()函數(shù)為例,描述了libuv處理這種File I/O請(qǐng)求時(shí)所涉及的各種操作:
建立thread pool(只建立一次) 在每個(gè)請(qǐng)求req__上綁定與其相關(guān)的event loop, work queue, work(), done() thread worker()用來(lái)處理work queue里面的每個(gè)請(qǐng)求,并執(zhí)行work() 通過(guò)uv_async_send()通知event loop執(zhí)行done()
以上就是關(guān)于本次相關(guān)的知識(shí)點(diǎn)內(nèi)容,感謝大家對(duì)腳本之家的支持。
- 虛函數(shù)表-C++多態(tài)的實(shí)現(xiàn)原理解析
- c++中new的三種用法詳細(xì)解析
- 淺析C++中結(jié)構(gòu)體的定義、初始化和引用
- c++ vector(向量)使用方法詳解(順序訪問(wèn)vector的多種方式)
- C++二叉樹結(jié)構(gòu)的建立與基本操作
- C++ 迷宮游戲?qū)崿F(xiàn)代碼
- C++生成dll和調(diào)用dll的方法實(shí)例
- C++實(shí)現(xiàn)簡(jiǎn)單的圖書管理系統(tǒng)
- 深入C++ string.find()函數(shù)的用法總結(jié)
- 淺析C/C++中sort函數(shù)的用法
- C++類靜態(tài)成員與類靜態(tài)成員函數(shù)詳解
- 詳解c++ libuv工作隊(duì)列
相關(guān)文章
node.js解決獲取圖片真實(shí)文件類型的問(wèn)題
這篇文章主要介紹了node.js解決獲取圖片真實(shí)文件類型的問(wèn)題,本文根據(jù)二進(jìn)制流及文件頭獲取文件類型mime-type,然后讀取文件二進(jìn)制的頭信息,獲取其真實(shí)的文件類型,需要的朋友可以參考下2014-12-12Node.js+ELK日志規(guī)范的實(shí)現(xiàn)
這篇文章主要介紹了Node.js+ELK日志規(guī)范的實(shí)現(xiàn),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2019-05-05詳解在Node.js中發(fā)起HTTP請(qǐng)求的5種方法
這篇文章主要介紹了詳解在Node.js中發(fā)起HTTP請(qǐng)求的5種方法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2019-01-01淺談Express.js解析Post數(shù)據(jù)類型的正確姿勢(shì)
這篇文章主要介紹了Express.js解析Post數(shù)據(jù)類型的正確姿勢(shì),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2019-05-05nodejs簡(jiǎn)單實(shí)現(xiàn)TCP服務(wù)器端和客戶端的聊天功能示例
這篇文章主要介紹了nodejs簡(jiǎn)單實(shí)現(xiàn)TCP服務(wù)器端和客戶端的聊天功能,結(jié)合實(shí)例形式分析了nodejs基于TCP協(xié)議實(shí)現(xiàn)的聊天程序客戶端與服務(wù)器端具體步驟與相關(guān)操作技巧,代碼備有較為詳盡的注釋便于理解,需要的朋友可以參考下2018-01-01安裝Node.js并啟動(dòng)本地服務(wù)的操作教程
今天小編就為大家分享一篇安裝Node.js并啟動(dòng)本地服務(wù)的操作教程,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-05-05使用Node.js實(shí)現(xiàn)一個(gè)文章生成器
本文將從零開始,講解如何使用Node.js來(lái)實(shí)現(xiàn)一個(gè)文章生成器,node里面有很多優(yōu)秀的模塊,現(xiàn)在我們就借助node的fs模塊來(lái)操控文本,來(lái)實(shí)現(xiàn)我們想要的效果,感興趣的小伙伴跟著小編一起來(lái)看看吧2024-07-07