1. 程式人生 > >NodeJS第1天筆記

NodeJS第1天筆記

左右 服務器優化 eat 平臺 減少 現在 客戶端連接 uri ive

Node.js入門和企業級應用開發實戰

目錄

一、Node.js簡介

1.1 簡介

1.2 特點

1.3 適合開發什麽?

1.4 Node.js無法挑戰老牌3P

二、Node.js安裝

三、HTTP模塊

四、js中的var、 let、const的區別:

一、Node.js簡介

1.1 簡介

V8引擎本身就是用於Chrome瀏覽器的JS解釋部分,但是Ryan Dahl這哥們,鬼才般的,把這個V8搬到了服務器上,用於做服務器的軟件。

Node.js是一個專註於實現高性能Web服務器優化的專家,幾經探索,幾經挫折後,遇到V8而誕生的項目。

Node.js是一個讓JavaScript運行在服務器端的開發平臺,它讓JavaScript的觸角伸到了服務器端,可以與PHP、JSP、Python、Ruby平起平坐。

但Node似乎有點不同:

Node.js不是一種獨立的語言,與PHP、JSP、Python、Perl、Ruby的"既是語言,也是平臺"不同,Node.js的使用JavaScript進行編程,運行在JavaScript引擎上(V8)。

● 與PHP、JSP等相比(PHP、JSP、.net都需要運行在服務器程序上,Apache、Naginx、Tomcat、IIS。

),Node.js跳過了Apache、Naginx、IIS等HTTP服務器,它自己不用建設在任何服務器軟件之上

。Node.js的許多設計理念與經典架構(LAMP = Linux + Apache + MySQL + PHP)有著很大的不同,可以提供強大的伸縮能力。一會兒我們就將看到,Node.js沒有web容器。

Node.js自身哲學,是花最小的硬件成本,追求更高的並發,更高的處理性能。

官網:https://nodejs.org/en/

特點:Node.js uses an event-driven, non-blocking I/O model that makes it lightweight and efficient.

1.2 特點

所謂的特點,就是Node.js是如何解決服務器高性能瓶頸問題的。

單線程

在Java、PHP或者.net等服務器端語言中,會為每一個客戶端連接創建一個新的線程。而每個線程需要耗費大約2MB內存。也就是說,理論上,一個8GB內存的服務器可以同時連接的最大用戶數為4000個左右。要讓Web應用程序支持更多的用戶,就需要增加服務器的數量,而Web應用程序的硬件成本當然就上升了。

Node.js不為每個客戶連接創建一個新的線程,而僅僅使用一個線程。當有用戶連接了,就觸發一個內部事件,通過非阻塞I/O、事件驅動機制,讓Node.js程序宏觀上也是並行的。使用Node.js,一個8GB內存的服務器,可以同時處理超過4萬用戶的連接。

另外,帶線程的帶來的好處,還有操作系統完全不再有線程創建、銷毀的時間開銷。

壞處,就是一個用戶造成了線程的崩潰,整個服務都崩潰了,其他人也崩潰了。

多線程、單線程的一個對比。

也就是說,單線程也能造成宏觀上的"並發"。

非阻塞I/O non-blocking I/O

例如,當在訪問數據庫取得數據的時候,需要一段時間。在傳統的單線程處理機制中,在執行了訪問數據庫代碼之後,整個線程都將暫停下來,等待數據庫返回結果,才能執行後面的代碼。也就是說,I/O阻塞了代碼的執行,極大地降低了程序的執行效率。

由於Node.js中采用了非阻塞型I/O機制,因此在執行了訪問數據庫的代碼之後,將立即轉而執行其後面的代碼,把數據庫返回結果的處理代碼放在回調函數中,從而提高了程序的執行效率。

當某個I/O執行完畢時,將以事件的形式通知執行I/O操作的線程,線程執行這個事件的回調函數。為了處理異步I/O,線程必須有事件循環,不斷的檢查有沒有未處理的事件,依次予以處理。

阻塞模式下,一個線程只能處理一項任務,要想提高吞吐量必須通過多線程。而非阻塞模式下,一個線程永遠在執行計算操作,這個線程的CPU核心利用率永遠是100%。所以,這是一種特別有哲理的解決方案:與其人多,但是好多人閑著;還不如一個人玩命,往死裏幹活兒。

事件驅動event-driven

在Node中,客戶端請求建立連接,提交數據等行為,會觸發相應的事件。在Node中,在一個時刻,只能執行一個事件回調函數,但是在執行一個事件回調函數的中途,可以轉而處理其他事件(比如,又有新用戶連接了),然後返回繼續執行原事件的回調函數,這種處理機制,稱為"事件環"機制。

Node.js底層是C++(V8也是C++寫的)。底層代碼中,近半數都用於事件隊列、回調函數隊列的構建。用事件驅動來完成服務器的任務調度,這是鬼才才能想到的。針尖上的舞蹈,用一個線程,擔負起了處理非常多的任務的使命。

單線程,單線程的好處,減少了內存開銷,操作系統的內存換頁。

如果某一個事情,進入了,但是被I/O阻塞了,所以這個線程就阻塞了。

非阻塞I/O, 不會傻等I/O語句結束,而會執行後面的語句。

非阻塞就能解決問題了麽?比如執行著小紅的業務,執行過程中,小剛的I/O回調完成了,此時怎麽辦??

事件機制,事件環,不管是新用戶的請求,還是老用戶的I/O完成,都將以事件方式加入事件環,等待調度。

說是三個特點,實際上是一個特點,離開誰都不行,都玩兒不轉了。

Node.js很像摳門的餐廳老板,只聘請1個服務員,服務很多人。結果,比很多服務員效率還高。

Node.js中所有的I/O都是異步的,回調函數,套回調函數。

1.3 適合開發什麽?

Node.js適合用來開發什麽樣的應用程序呢?

善於I/O,不善於計算。因為Node.js最擅長的就是任務調度,如果你的業務有很多的CPU計算,實際上也相當於這個計算阻塞了這個單線程,就不適合Node開發。

當應用程序需要處理大量並發的I/O,而在向客戶端發出響應之前,應用程序內部並不需要進行非常復雜的處理的時候,Node.js非常適合。Node.js也非常適合與web socket配合,開發長連接的實時交互應用程序。

比如:

● 用戶表單收集

● 考試系統

● 聊天室

● 圖文直播

● 提供JSON的API(為前臺Angular使用)

1.4 Node.js無法挑戰老牌3P

二、Node.js安裝

Node.js和Java非常像,跨平臺的。不管是Linux還是windows編程是完全一致的(有一些不一樣,比如路徑的表述)。Linux版本的Node.js環境和windows環境是不一樣的,但是編程語言一樣。很像Java虛擬機。

我們學習的是最新版,2015年9月19日最新版本4.1.0。

安裝包的大小只有10M不到,真的就是一個小玩具。

裝完之後,我們在系統的環境變量裏面,我們看一下:

環境變量,就已經自動的填寫進去了,就是我們node安裝的目錄。

什麽叫做環境變量?就是在系統的任何目錄下,都能運行c:\program files\nodejs裏面的程序。

在cmd中,輸入node -v就能夠查看nodejs的版本。你會發現,我們現在的盤符,不在安裝目錄下,但是也能夠運行,這就是因為有系統環境變量。系統的環境變量已經有了c:\program files\nodejs了,所以,這個文件夾中的node.exe就能夠在任何盤符運行。

運行文件,就要用node命令來運行:

node 路徑名

推薦,不要使用完整的路徑名,而是通過先進入案例文件夾,然後node相對地址

cd命令,就是change directory的縮寫,表示更改當前目錄。

Node.js是服務器的程序,寫的js語句,都將運行在服務器上。返回給客戶的,都是已經處理好的純html。

//require表示引包,引包就是引用自己的一個特殊功能

var http = require("http");

//創建服務器,參數是一個回調函數,表示如果有請求進來,要做什麽

var server = http.createServer(function(req,res){

//req表示請求,request; res表示響應,response

//設置HTTP頭部,狀態碼是200,文件類型是html,字符集是utf8

res.writeHead(200,{"Content-type":"text/html;charset=UTF-8"});

res.end("哈哈哈哈,我買了一個iPhone" + (1+2+3) + "s");

});

//運行服務器,監聽3000端口(端口號可以任改)

server.listen(3000,"127.0.0.1");

如果想修改程序,必須中斷當前運行的服務器,重新node一次,刷新,才行。

ctrl+c,就可以打斷掛起的服務器程序。此時按上箭頭,能夠快速調用最近的node命令。

你會發現,我們本地寫一個js,打死都不能直接拖入瀏覽器運行,但是有了node,我們任何一個js文件,都可以通過node來運行。也就是說,node就是一個js的執行環境。

我們現在,要跑起來一個服務器,這個服務器的腳本,要以.js存儲。是一個js文件。用node命令運行這個js文件罷了。

Node.js沒有根目錄的概念,因為它根本沒有任何的web容器!

讓node.js提供一個靜態服務,都非常難!

也就是說,node.js中,如果看見一個網址是

127.0.0.1:3000/fang

別再去想,一定有一個文件夾,叫做fang了。可能/fang的物理文件,是同目錄的test.html

URL和真實物理文件,是沒有關系的。URL是通過了Node的頂層路由設計,呈遞某一個靜態文件的。

三、HTTP模塊

Node.js中,將很多的功能,劃分為了一個個mudule,大陸的書翻譯為模塊;臺灣的書,翻譯為模組。

這是因為,有一些程序需要使用fs功能(文件讀取功能),有一些不用的,所以為了效率,你用啥,你就require啥。

//這個案例簡單講解http模塊

//引用模塊

var http = require("http");

//創建一個服務器,回調函數表示接收到請求之後做的事情

var server = http.createServer(function(req,res){

//req參數表示請求,res表示響應

console.log("服務器接收到了請求" + req.url);

res.end();

});

//監聽端口

server.listen(3000,"127.0.0.1");

設置一個響應頭:

res.writeHead(200,{"Content-Type":"text/plain;charset=UTF8"});

我們現在來看一下req裏面能夠使用的東西。

最關鍵的就是req.url屬性,表示用戶的請求URL地址。所有的路由設計,都是通過req.url來實現的。

我們比較關心的不是拿到URL,而是識別這個URL。

識別URL,用到兩個新模塊,第一個就是url模塊,第二個就是querystring模塊

字符串查詢,用querystring處理

querystring.parse(‘foo=bar&baz=qux&baz=quux&corge‘)

// returns

{ foo: ‘bar‘, baz: [‘qux‘, ‘quux‘], corge: ‘‘ }

// Suppose gbkDecodeURIComponent function already exists,

// it can decode `gbk` encoding string

querystring.parse(‘w=%D6%D0%CE%C4&foo=bar‘, null, null,

{ decodeURIComponent: gbkDecodeURIComponent })

// returns

{ w: ‘中文‘, foo: ‘bar‘ }

四、js中的var、 let、const的區別:

技術分享圖片

NodeJS第1天筆記