node.js是什么?
這是一種通過JavaScript語言開發(fā)web服務(wù)端的東西”。
為什么我們要用node.js?
node.js有非阻塞,事件驅(qū)動I/O等特性,從而讓高并發(fā)(high concurrency)在的輪詢(Polling)和comet構(gòu)建的應(yīng)用中成為可能。
瀏覽器給網(wǎng)站發(fā)請求的過程一直沒怎么變過。當瀏覽器給網(wǎng)站發(fā)了請求。服務(wù)器收到了請求,然后開始搜尋被請求的資源。如果有需要,服務(wù)器還會查詢一下數(shù)據(jù)庫,最后把響應(yīng)結(jié)果傳回瀏覽器。不過,在傳統(tǒng)的web服務(wù)器中(比如Apache),每一個請求都會讓服務(wù)器創(chuàng)建一個新的進程來處理這個請求。后來有了Ajax。有了Ajax,我們就不用每次都請求一個完整的新頁面了,取而代之的是,每次只請求需要的部分頁面信息就可以了。這顯然是一個進步。但是比如你要建一個FriendFeed這樣的社交網(wǎng)站(類似人人網(wǎng)那樣的刷朋友新鮮事的網(wǎng)站),你的好友會隨時的推送新的狀態(tài),然后你的新鮮事會實時自動刷新。要達成這個需求,我們需要讓用戶一直與服務(wù)器保持一個有效連接。目前最簡單的實現(xiàn)方法,就是讓用戶和服務(wù)器之間保持長輪詢(long polling)。HTTP請求不是持續(xù)的連接,你請求一次,服務(wù)器響應(yīng)一次,然后就完了。長輪訓(xùn)是一種利用HTTP模擬持續(xù)連接的技巧。具體來說,只要頁面載入了,不管你需不需要服務(wù)器給你響應(yīng)信息,你都會給服務(wù)器發(fā)一個Ajax請求。這個請求不同于一般的Ajax請求,服務(wù)器不會直接給你返回信息,而是它要等著,直到服務(wù)器覺得該給你發(fā)信息了,它才會響應(yīng)。比如,你的好友發(fā)了一條新鮮事,服務(wù)器就會把這個新鮮事當做響應(yīng)發(fā)給你的瀏覽器,然后你的瀏覽器就刷新頁面了。瀏覽器收到響應(yīng)刷新完之后,再發(fā)送一條新的請求給服務(wù)器,這個請求依然不會立即被響應(yīng)。于是就開始重復(fù)以上步驟。利用這個方法,可以讓瀏覽器始終保持等待響應(yīng)的狀態(tài)。雖然以上過程依然只有非持續(xù)的Http參與,但是我們模擬出了一個看似持續(xù)的連接狀態(tài)我們再看傳統(tǒng)的服務(wù)器(比如Apache)。每次一個新用戶連到你的網(wǎng)站上,你的服務(wù)器就得開一個連接。每個連接都需要占一個進程,這些進程大部分時間都是閑著的(比如等著你好友發(fā)新鮮事,等好友發(fā)完才給用戶響應(yīng)信息。或者等著數(shù)據(jù)庫返回查詢結(jié)果什么的)。雖然這些進程閑著,但是照樣占用內(nèi)存。這意味著,如果用戶連接數(shù)的增長到一定規(guī)模,你服務(wù)器沒準就要耗光內(nèi)存直接癱了。這種情況怎么解決?解決方法就是剛才上邊說的:非阻塞和事件驅(qū)動。這些概念在我們談的這個情景里面其實沒那么難理解。你把非阻塞的服務(wù)器想象成一個loop循環(huán),這個loop會一直跑下去。一個新請求來了,這個loop就接了這個請求,把這個請求傳給其他的進程(比如傳給一個搞數(shù)據(jù)庫查詢的進程),然后響應(yīng)一個回調(diào)(callback)。完事了這loop就接著跑,接其他的請求。這樣下來。服務(wù)器就不會像之前那樣傻等著數(shù)據(jù)庫返回結(jié)果了。如果數(shù)據(jù)庫把結(jié)果返回來了,loop就把結(jié)果傳回用戶的瀏覽器,接著繼續(xù)跑。在這種方式下,你的服務(wù)器的進程就不會閑著等著。從而在理論上說,同一時刻的數(shù)據(jù)庫查詢數(shù)量,以及用戶的請求數(shù)量就沒有限制了。服務(wù)器只在用戶那邊有事件發(fā)生的時候才響應(yīng),這就是事件驅(qū)動。FriendFeed是用基于Python的非阻塞框架Tornado (知乎也用了這個框架) 來實現(xiàn)上面說的新鮮事功能的。不過,Node.js就比前者更妙了。Node.js的應(yīng)用是通過javascript開發(fā)的,然后直接在Google的變態(tài)V8引擎上跑。用了Node.js,你就不用擔心用戶端的請求會在服務(wù)器里跑了一段能夠造成阻塞的代碼了。因為javascript本身就是事件驅(qū)動的腳本語言。你回想一下,在給前端寫javascript的時候,更多時候你都是在搞事件處理和回調(diào)函數(shù)。javascript本身就是給事件處理量身定制的語言。Node.js還是處于初期階段。如果你想開發(fā)一個基于Node.js的應(yīng)用,你應(yīng)該會需要寫一些很底層代碼。但是下一代瀏覽器很快就要采用WebSocket技術(shù)了,從而長輪詢也會消失。在Web開發(fā)里,Node.js這種類型的技術(shù)只會變得越來越重要。