理解 Node.js
當(dāng)我把Node.js介紹給人們的時(shí)候,通常會(huì)產(chǎn)生兩種反應(yīng)。總的來(lái)說(shuō)人們不是可以正確理解它,就是被它搞得非常糊涂。
如果你到目前為止還處于第二種,下面是我想要用來(lái)講解node的內(nèi)容:
- 它是一個(gè)命令行工具。你下載一個(gè)tarball文件,編譯并安裝源碼。
- 它可以讓你在你的終端中通過(guò)輸入’node my_app.js’來(lái)運(yùn)行JavaScript程序。
- JS是被V8 javascript引擎(使Google Chrome運(yùn)行如此之快的東西)所執(zhí)行的。
- Node提供了一個(gè)用來(lái)訪問(wèn)網(wǎng)絡(luò)和文件系統(tǒng)的JavaScript API。
“但是我可以做任何我想要做的事情用:ruby,python,php,java, … !”
我聽(tīng)到了你的聲音。你是對(duì)的。Node不是該死的獨(dú)角獸,它不會(huì)幫你做你自己的工作,抱歉。它只是一個(gè)工具,并且它大概也不會(huì)完全地替換你常用的工具,至少當(dāng)前不會(huì)。
“說(shuō)重點(diǎn)!”
好的,我會(huì)的。Node總的來(lái)說(shuō)非常優(yōu)秀當(dāng)你需要同時(shí)做許多事情的時(shí)候。你有過(guò)寫(xiě)了一部分代碼然后說(shuō)“我想要這個(gè)并行執(zhí)行”嗎?好吧,在node中任何東西都是并行運(yùn)行的,除了你的代碼。
“嘿?”
是的沒(méi)錯(cuò),任何東西都是并行的,除了你的代碼。為了理解它,把你的代碼想像成是國(guó)王,然后node是它的仆從軍隊(duì)。
新的一天以仆從叫醒國(guó)王并且問(wèn)他是否需要什么而開(kāi)始了。國(guó)王給了這個(gè)仆從一個(gè)任務(wù)列表,然后回去繼續(xù)睡覺(jué)去了(汗)。這個(gè)仆從把這些任務(wù)分配給了他的同僚,然后他們開(kāi)始工作。
當(dāng)一個(gè)仆從完成了一個(gè)任務(wù),他會(huì)在國(guó)王領(lǐng)地外邊排成一條線來(lái)匯報(bào)。國(guó)王一次讓一個(gè)仆從進(jìn)來(lái),然后聽(tīng)取他的報(bào)告。有時(shí)候國(guó)王會(huì)在仆從出去的時(shí)候給仆從更多的任務(wù)。
生活是美好的,因?yàn)閲?guó)王的仆從并行的執(zhí)行他的任務(wù),但是一次只報(bào)告一個(gè)結(jié)果,所以國(guó)王可以專注。*
“那是異想天開(kāi),但是你能結(jié)束這個(gè)愚蠢的比喻并且用geek的方式告訴我嗎?”
當(dāng)然。一個(gè)簡(jiǎn)單的node程序看起來(lái)可能是這樣的:
- <code>var fs = require('fs')
- , sys = require('sys');
- fs.readFile('treasure-chamber-report.txt', function(report) {
- sys.puts("oh, look at all my money: "+report);
- });
- fs.writeFile('letter-to-princess.txt', '...', function() {
- sys.puts("can't wait to hear back from her!");
- });
- </code>
你的代碼給了node兩個(gè)任務(wù)用來(lái)讀寫(xiě)一個(gè)文件,然后就休眠了。當(dāng)node完成了一個(gè)任務(wù),跟它對(duì)應(yīng)的回調(diào)就會(huì)被觸發(fā)。但是在同一時(shí)間只能有一個(gè)回調(diào)觸發(fā)。在那個(gè)回調(diào)完成了執(zhí)行之前,所有其他的回調(diào)不得不排隊(duì)等待。進(jìn)一步說(shuō),對(duì)于回調(diào)觸發(fā)的順序是無(wú)法保證的。
“所以我不必?fù)?dān)心代碼在同一時(shí)間訪問(wèn)同一個(gè)數(shù)據(jù)結(jié)構(gòu)?”
你理解了!這就是JavaScript的單進(jìn)程/事件循環(huán)設(shè)計(jì)的全部美之所在。
“非常好,但是為什么我應(yīng)該用它?”
一個(gè)原因是效率。在一個(gè)web應(yīng)用中,你主要的響應(yīng)時(shí)間消耗通常是執(zhí)行數(shù)據(jù)庫(kù)查詢的時(shí)間之和。通過(guò)node,你可以一次執(zhí)行你所有的查詢,把響應(yīng)時(shí)間減少為執(zhí)行最慢查詢所花費(fèi)的時(shí)間。
另一個(gè)原因是JavaScript。你可以使用node來(lái)在瀏覽器和你的后端之間共享代碼。JavaScript也正在變成一個(gè)真的通用語(yǔ)言。不管過(guò)去你用的是python,ruby,java,php或者其他語(yǔ)言,你都可能用過(guò)一些JS,對(duì)嗎?
***一個(gè)原因是原始速度。V8時(shí)刻在向成為全球最快的動(dòng)態(tài)語(yǔ)言解釋器之一而努力。我想不到任何其他語(yǔ)言在速度上有像如今的JavaScript一樣有如此突飛猛進(jìn)的提升。進(jìn)一步說(shuō),node的I/O能力非常輕量級(jí),可以使你可以盡可能的完全利用你系統(tǒng)完全的I/O能力。
“所以你在說(shuō)我應(yīng)該從現(xiàn)在起用node來(lái)寫(xiě)我所有的應(yīng)用?”
是和不是。一旦你開(kāi)始掄node這個(gè)錘子,那么顯然一切開(kāi)始看起來(lái)都像一個(gè)釘子。但是如果你當(dāng)前的工作有一個(gè)期限,你可以通過(guò)以下來(lái)決定:
- 是否低響應(yīng)時(shí)間/高并發(fā)重要?Node真的很擅長(zhǎng)它。
- 項(xiàng)目有多大?小項(xiàng)目問(wèn)題不大。大項(xiàng)目應(yīng)該小心的評(píng)估(可用的庫(kù),修復(fù)一個(gè)bug所需的資源或者two upstream等等。)。
“node能運(yùn)行在Windows上嗎?”
不行。如果你使用的是windows,你需要運(yùn)行一個(gè)Linux虛擬機(jī)(我推薦VirtualBox)。node對(duì)Windows的支持在計(jì)劃中了,但是接下來(lái)的幾個(gè)月不要屏住呼吸除非你想對(duì)port提供幫助。(譯者:現(xiàn)在node可以***的運(yùn)行在Windows上)
“我能在node中訪問(wèn)DOM嗎?”
好問(wèn)題!不行,DOM是瀏覽器中的東東,并且node的JS引擎(V8)幸好跟那些混亂的東西是完全分離的。不過(guò),有人在以node模塊的形式來(lái)實(shí)現(xiàn)DOM,可能帶來(lái)令人興奮的可能性比如對(duì)客戶端代碼進(jìn)行單元測(cè)試。
“難道事件驅(qū)動(dòng)編程不是真的非常難嗎?”
這取決于你。如果你已經(jīng)學(xué)過(guò)如何在瀏覽器中擺弄AJAX調(diào)用和用戶事件,那么使用node不會(huì)是一個(gè)問(wèn)題。
同時(shí),測(cè)試驅(qū)動(dòng)開(kāi)發(fā)可以真正的幫助你以一個(gè)可維護(hù)的設(shè)計(jì)做為開(kāi)始。
“誰(shuí)在用它?”
在node wiki(滾動(dòng)到”Companies using Node”)有一個(gè)小的/不全的列表。Yahoo正在為YUI對(duì)node進(jìn)行實(shí)驗(yàn),Plurk正在使用它處理大規(guī)模的comet,并且Paul Bakaus(由于jQuery UI而出名)正在創(chuàng)建一個(gè)令人興奮的游戲引擎,其中后端使用了一些node代碼。Joyent已經(jīng)雇傭了Ryan Dahl(node作者)并且大力資助開(kāi)發(fā)。
對(duì)了,Heroku也剛剛聲明了支持對(duì)node.js的托管。
“我能去哪里學(xué)更多?”
Tim Caswell正在運(yùn)作優(yōu)秀的How To Node博客。在twitter上Follow #nodejs。訂閱郵件列表。然后去IRC頻道逛逛,#node.js(是的,名字中包含這個(gè)點(diǎn))。我們?cè)谀堑膭澩Х謹(jǐn)?shù)快達(dá)到200了:)。
我也將會(huì)繼續(xù)在debuggable.com這里寫(xiě)文章。
當(dāng)前就寫(xiě)這么多了。如果你有其他問(wèn)題歡迎留言。
–注解:
*: 這個(gè)比喻顯然是太簡(jiǎn)單了,但是在現(xiàn)實(shí)之中找到一個(gè)與非阻塞概念相對(duì)應(yīng)的對(duì)照物很難。
原文鏈接:http://debuggable.com/posts/understanding-node-js:4bd98440-45e4-4a9a-8ef7-0f7ecbdd56cb