node核心特性理解

原文地址在我的博客,轉(zhuǎn)載請注明來源,謝謝!

node是在前端領(lǐng)域經(jīng)??吹降脑~。node對于前端的重要性已經(jīng)不言而喻,掌握node也是作為合格的前端工程師一項基本功了。知道node、知道后端的一些東西,才能更好的與別人合作,發(fā)揮更大的價值。

概述

本文主要介紹了我對node的一些核心特性的理解,包括node架構(gòu)、特點、機制、核心模塊與簡單應(yīng)用。

正文

從瀏覽器到node

首先,node是一個平臺,使用javascript作為編程語言,運行在服務(wù)端。服務(wù)端語言能做的,node一般都能做,而且有些情況下做的更好,因為它具有自己的特色。

node是javascript運行環(huán)境(runtime),就像瀏覽器一樣,是一個平臺。在瀏覽器中,V8引擎負責(zé)解釋javascript,你在javascript調(diào)用的接口都是瀏覽器實現(xiàn)并提供的,瀏覽器會調(diào)用底層的、由其他語言(C++)實現(xiàn)并封裝好的接口來完成任務(wù);同樣,在node中,也是V8引擎負責(zé)解釋javascript,而你在javascript調(diào)用的瀏覽器提供的接口就不能用了,因為它脫離了瀏覽器的環(huán)境,但是因為你在node環(huán)境中,你就可以使用node提供的由C++語言實現(xiàn)的、由javascript封裝好的各種接口來完成后端任務(wù)。瀏覽器提供的API用于處理前端任務(wù),比如彈個窗,換個主題,處理用戶操作等,而node因為服務(wù)后端,因此提供的API則用來處理后端任務(wù),比如響應(yīng)請求,讀取文件等,這些API由不同的模塊提供。因為關(guān)注領(lǐng)域不一樣,因此所做的任務(wù)就不一樣,提供的API就不一樣,但是原理、相關(guān)實現(xiàn)大致與瀏覽器端相同。

從瀏覽器到node這一塊如果想了解更多,推薦IBM的文章node.js到底是什么?

node 架構(gòu)

node架構(gòu)分為三層(參考鏈接):

圖片來源

  • Node standard library:node標(biāo)準(zhǔn)庫,也就是node模塊提供各種接口的javascript實現(xiàn),任何javascript代碼、npm install 或者你寫的模塊都在這里
  • Node bindings:包括C/C++ bindings(膠水代碼)和Add on(添加其他C/C++庫時需要自己寫的Bindings),這一層向下封裝了V8和libuv接口,向上提供了基礎(chǔ)API接口,是連接javascript和C++的橋梁
  • 第三層是支撐 Node.js 運行的關(guān)鍵,由 C/C++ 實現(xiàn)。
    • V8 是Google開發(fā)的JavaScript引擎,提供JavaScript運行環(huán)境,可以說它就是 Node.js 的發(fā)動機,負責(zé)解釋javascript,與chrome瀏覽器相同。
    • Libuv 是專門為Node.js開發(fā)的一個封裝庫,提供跨平臺的異步I/O能力,負責(zé)node運行時的線程池調(diào)度。
    • C-ares:提供了異步處理 DNS 相關(guān)的能力。
    • http_parser、OpenSSL、zlib 等:提供包括 http 解析、SSL、數(shù)據(jù)壓縮等系統(tǒng)底層的訪問。

平常我們用到的也就是第一層node各個模塊實現(xiàn)的接口。

那他們之間時如何協(xié)作的呢

javascript主線程

程序啟動,V8引擎會首先解析javascript代碼,通過Node bindings來調(diào)用C/C++庫。執(zhí)行到當(dāng)前事件時,會把事件放在調(diào)用堆棧(stack和heap)處理(可以理解為放進一個工作空間,如上圖),在堆棧中的任何I/O請求都會交給libuv來處理,libuv維持一個線程池,里面是一些工作線程(如下圖),請求會調(diào)用這些線程來完成任務(wù),這些線程則調(diào)用底層的C/C++庫。完成時,libuv再把結(jié)果返回事件隊列等待主線程執(zhí)行。在此期間,主線程繼續(xù)執(zhí)行其他任務(wù)。

node 執(zhí)行特性

單線程、非阻塞型I/O

單線程的意思就是只在一個線程上運行javascript。首先,javascript 在瀏覽器端是單線程的,這是為了避免多線程產(chǎn)生任務(wù)沖突的情況;其次,java和PHP這類多線程后端語言,為避免同步I/O阻塞,每處理一個連接都會產(chǎn)生一個新線程,這樣的話在遇到大量并發(fā)請求時就會受到物理內(nèi)存的限制。node 延續(xù)了瀏覽器端單線程javascript,只用一個主線程執(zhí)行javascript,不斷循環(huán)遍歷事件隊列,執(zhí)行事件。事實上,主線程發(fā)出的I/O請求,都會交給其他線程去完成,其他線程完成后悔返回結(jié)果放到事件隊列。在此期間,主線程會繼續(xù)執(zhí)行其他任務(wù),也就是在交給libuv后直接返回,繼續(xù)執(zhí)行下面的任務(wù),主線程只負責(zé)循環(huán)執(zhí)行事件隊列,因此這種模式稱為非阻塞型I/O,性能很好,適用于處理大量并發(fā)請求,還能簡化開發(fā)。

事件驅(qū)動機制

還是跟瀏覽器的差不多。總的來說就是,瀏覽器端把鼠標(biāo)點擊、鍵盤按鍵等定義為事件,而node把網(wǎng)絡(luò)請求、I/O操作等也看作事件,嚴(yán)格來說,一切動作都是事件,這就是事件驅(qū)動的思想。在程序啟動時,便進入事件循環(huán),不斷遍歷執(zhí)行事件隊列中產(chǎn)生的事件,而在執(zhí)行過程中,又會產(chǎn)生新的事件,因此稱為事件循環(huán)。主線程執(zhí)行事件時,遇到麻煩的I/O請求會交給libuv來調(diào)度其他工作線程來幫忙,忙完后就會形成事件返回結(jié)果到事件隊列等待主線程處理。在此期間,主線程會繼續(xù)執(zhí)行其他任務(wù)。

mbp 曾經(jīng)做過一個巧妙的比喻,把 Node.js 看成一家餐廳。我在此借用下他的例子,稍作修改來闡述下 Node.js 的執(zhí)行情況:

把 Node.js 應(yīng)用程序想象成一家星巴克,一個訓(xùn)練有素的前臺服務(wù)生(唯一的主線程)在柜臺前接受訂單。當(dāng)很多顧客同時光臨的時候,他們排隊(進入事件隊列)等候接待;每當(dāng)服務(wù)生接待一位顧客,服務(wù)生會把訂單告知給經(jīng)理(libuv),經(jīng)理安排相應(yīng)的專職人員去烹制咖啡(工作線程或者系統(tǒng)特性)。這個專職人員會使用不同的原料和咖啡機(底層 C/C++ 組件)按訂單要求制作咖啡或甜點,通常會有四個這樣的專職人員保持在崗待命(線程池),高峰期的時候也可以安排更多(不過需要在一早就安排人員來上班,而不能中午臨時通知)。服務(wù)生把訂單轉(zhuǎn)交給經(jīng)理之后不需要等著咖啡制作完成,而是直接開始接待下一位顧客(事件循環(huán)放進調(diào)用堆棧的另一個事件),你可以把當(dāng)前調(diào)用堆棧里的事件看成是站在柜臺前正在接受服務(wù)的顧客。

當(dāng)咖啡完成時,會被發(fā)送到顧客隊列的最后位置,等它移動到柜臺前服務(wù)生會叫相應(yīng)顧客的名字,顧客就來取走咖啡(最后這部分在真實生活中聽起來有點怪,不過你從程序執(zhí)行的角度理解就比較合乎情理了)。

? ——By Amio

如果你想進一步了解javascript 事件驅(qū)動機制,推薦深入理解 javascript 事件循環(huán)機制

node 模塊

node 模塊機制是CommonJs 的實現(xiàn)。起初,javascript 標(biāo)準(zhǔn)一片混沌,并沒有其他成熟語言(例如C++)的模塊機制、標(biāo)準(zhǔn)庫、接口等,為了讓javascript 具備開發(fā)大型應(yīng)用的能力,為了讓 javascript 能在后端運行,CommonJS 就制定了javascript 模塊規(guī)范。node 借鑒了這個規(guī)范,讓javascript 以模塊形式組織起來。模塊機制是一個成熟語言必備的,一個模塊代表一個功能的封裝,它就像搭積木一樣,不同模塊可以銜接在一塊,使語言具有極強的可擴展型。node 模塊機制同時制定了模塊規(guī)范,能讓全球的開發(fā)者都可以在node官網(wǎng)上傳自己的包。此外,node 社區(qū)又實現(xiàn)了node 包管理器npm,使用npm可以輕松管理各種包。

node 的模塊分為核心模塊和用戶模塊,前者是底層的、自帶的,后者是第三方。

核心模塊有Global(全局對象)、Http、fs(文件系統(tǒng))、Buffer、Stream、Events、URL、path等,這些模塊提供了后端服務(wù)的基本功能,都提供自己關(guān)注功能的API。

在使用模塊時,require 即可。但在require背后,node 有一套尋找模塊的機制:

node require機制

從上圖可以看到,node 優(yōu)先從緩存區(qū)讀取,緩存區(qū)有直接讀取,沒有則加載并緩存,這樣做不用一遍一遍去找了,非常高效。node 在緩存區(qū)沒有發(fā)現(xiàn)模塊時,會分析require 的路徑和文件后綴,node 有個模塊路徑的查找策略,我們可以在名為module_paths 的js文件里console.log(module.paths)然后node module_paths.js運行來間接查看node 尋找文件模塊的具體文件的方式:

[ '/home/username/nodeProject/node_modules',
 '/home/username/node_modules',
 '/home/node_modules',
 '/node_modules' ] //Linux下的數(shù)組輸出(/home/username因電腦不同而異)

[ 'c:\\nodeProject\\node_modules', 'c:\\node_modules' ] //Windows

也就是按照下面的順序:

  • 當(dāng)前文件目錄下的node_modules目錄。
  • 父目錄下的node_modules目錄。
  • 父目錄的父目錄下的node_modules目錄。
  • 沿路徑向上逐級遞歸,直到根目錄下的node_modules目錄。

這些順序都是在查找緩存之后的。

在找到模塊后,node 將在引入之前對這個模塊進行編譯執(zhí)行,編譯成功后會緩存,執(zhí)行的結(jié)果會返回給調(diào)用者。

簡單應(yīng)用

有了node 自帶核心模塊的基礎(chǔ)功能,就可以進一步封裝更強大、容易操作的功能了,就像jQuery 對于javascript 基礎(chǔ)API 一樣,node 社區(qū)也誕生了像 Express、KOA等框架來構(gòu)建node.js程序

node.js開發(fā)框架

這些框架的詳情移步2017 Node.js 開發(fā)框架比較

另外,node 還可以連接MySQL,MangoDB進行數(shù)據(jù)庫操作。

下面是使用express 腳手架生成的基本 node應(yīng)用結(jié)構(gòu):

.
├── app.js            //程序入口
├── bin
│   └── www           //二機制文件
├── package.json      //項目配置文件
├── public
│   ├── images        
│   ├── javascripts
│   └── stylesheets
│       └── style.css
├── routes
│   ├── index.js      //路由文件入口
│   └── users.js
└── views
    ├── error.jade    //界面模板
    ├── index.jade
    └── layout.jade

現(xiàn)在使用 node作為后端語言通常都要配合類庫和框架使用。

node 的單線程、非阻塞型特點讓它非常適合高并發(fā)的應(yīng)用,適合處理大量重復(fù)的、簡單的邏輯,適合構(gòu)建Rest/JSON API服務(wù);同時,也正是因為這些特性,node 不適合CPU使用率較重、IO使用率較輕的偏計算應(yīng)用。缺點是因為單線程,一個進程掛就全掛了,可靠性低,但這是可以避免的。node 更多的應(yīng)用是在前端、中間件、前后端分離等。

由于 node 的諸多優(yōu)點,現(xiàn)在越來越多大公司開始使用node、深度使用node。

總結(jié)

node 的核心概念、思想遠不止這么多,應(yīng)用更是多了去了,無奈本人水平有限,只能說個淺層,還有很多像進程管理、異步編程、異常調(diào)試、部署、性能調(diào)優(yōu)、與集群、CDN協(xié)調(diào)等都值得深入探索一下。無論如何,node 是讓javascript 邁向企業(yè)級開發(fā)語言重要的一步(也許已經(jīng)是了),前端工程師從未像現(xiàn)在這樣的powerful,能做的事情越來越多,所能涉及的領(lǐng)域也越來越多。前端這行越來越令人興奮了。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Node.js是目前非?;馃岬募夹g(shù),但是它的誕生經(jīng)歷卻很奇特。 眾所周知,在Netscape設(shè)計出JavaScri...
    w_zhuan閱讀 3,737評論 2 41
  • Node.js是目前非?;馃岬募夹g(shù),但是它的誕生經(jīng)歷卻很奇特。 眾所周知,在Netscape設(shè)計出JavaScri...
    Myselfyan閱讀 4,206評論 2 58
  • 人有許多路可以選,可以走。
    馬不蹄兒閱讀 382評論 0 0
  • “這么說,你老婆是南方人咯?”我緊接著問道。 “恩?!睂γ娴拇笫屐t腆的笑笑,被長年累月的油煙熏得油光滑亮的臉上綻開...
    美好的花想容閱讀 554評論 0 0
  • 黑夜中響起璀璨 恍若煙火明亮 一閃一響,此起彼伏 單調(diào)而又凄涼 這絕不是人間的光 那么刺眼又無情 靈魂被收割 發(fā)出...
    淺笑大大閱讀 358評論 0 0

友情鏈接更多精彩內(nèi)容