node.js學習筆記

什麼是node.js?node

簡單的說 Node.js 就是運行在服務端的 JavaScript。web

Node.js 是一個基於Chrome JavaScript 運行時創建的一個平臺。shell

Node.js是一個事件驅動I/O服務端JavaScript環境,基於Google的V8引擎,V8引擎執行Javascript的速度很是快,性能很是好。編程

 

應用包括那幾個部分?設計模式

使用 Node.js 時,咱們不單單 在實現一個應用,同時還實現了整個 HTTP 服務器。事實上,咱們的 Web 應用以及對應的 Web 服務器基本上是同樣的。瀏覽器

在咱們建立 Node.js 第一個 "Hello, World!" 應用前,讓咱們先了解下 Node.js 應用是由哪幾部分組成的:服務器

  1. 引入 required 模塊:咱們可使用 require 指令來載入 Node.js 模塊。併發

  2. 建立服務器:服務器能夠監聽客戶端的請求,相似於 Apache 、Nginx 等 HTTP 服務器。異步

  3. 接收請求與響應請求 服務器很容易建立,客戶端可使用瀏覽器或終端發送 HTTP 請求,服務器接收請求後返回響應數據。異步編程

 

步驟:

一、引入required模塊,咱們使用 require 指令來載入 http 模塊,並將實例化的 HTTP 賦值給變量 http,實例以下:

var http = require("http");

二、建立服務器

接下來咱們使用 http.createServer() 方法建立服務器,並使用 listen 方法綁定 8888 端口。 函數經過 request, response 參數來接收和響應數據。

實例以下,在你項目的根目錄下建立一個叫 server.js 的文件,並寫入如下代碼:

var http = require('http');

http.createServer(function (request, response) {

    // 發送 HTTP 頭部 
    // HTTP 狀態值: 200 : OK
    // 內容類型: text/plain
    response.writeHead(200, {'Content-Type': 'text/plain'});

    // 發送響應數據 "Hello World"
    response.end('Hello World\n');
}).listen(8888);

// 終端打印以下信息
console.log('Server running at http://127.0.0.1:8888/');

以上代碼咱們完成了一個能夠工做的 HTTP 服務器。

使用 node 命令執行以上的代碼:

node server.js
Server running at http://127.0.0.1:8888/

分析Node.js 的 HTTP 服務器:

  • 第一行請求(require)Node.js 自帶的 http 模塊,而且把它賦值給 http 變量。
  • 接下來咱們調用 http 模塊提供的函數: createServer 。這個函數會返回 一個對象,這個對象有一個叫作 listen 的方法,這個方法有一個數值參數, 指定這個 HTTP 服務器監聽的端口號。

 

Node.js REPL(交互式解釋器)

Node.js REPL(Read Eval Print Loop:交互式解釋器) 表示一個電腦的環境,相似 Window 系統的終端或 Unix/Linux shell,咱們能夠在終端中輸入命令,並接收系統的響應。

REPL 命令

  • ctrl + c - 退出當前終端。

  • ctrl + c 按下兩次 - 退出 Node REPL。

  • ctrl + d - 退出 Node REPL.

  • 向上/向下 鍵 - 查看輸入的歷史命令

  • tab 鍵 - 列出當前命令

  • .help - 列出使用命令

  • .break - 退出多行表達式

  • .clear - 退出多行表達式

  • .save filename - 保存當前的 Node REPL 會話到指定文件

  • .load filename - 載入當前 Node REPL 會話的文件內容。

node.js回調函數

Node.js 異步編程的直接體現就是回調。

異步編程依託於回調來實現,但不能說使用了回調後程序就異步化了。

回調函數在完成任務後就會被調用,Node 使用了大量的回調函數,Node 全部 API 都支持回調函數。

例如,咱們能夠一邊讀取文件,一邊執行其餘命令,在文件讀取完成後,咱們將文件內容做爲回調函數的參數返回。這樣在執行代碼時就沒有阻塞或等待文件 I/O 操做。這就大大提升了 Node.js 的性能,能夠處理大量的併發請求。

阻塞代碼實例:

建立一個txt文件:input.txt,內容是'hello world';

建立main.js:

var fs = require("fs");

var data = fs.readFileSync('input.txt');

console.log(data.toString());
console.log("程序執行結束!");

執行結果:

菜鳥教程官網地址:www.runoob.com 程序執行結束!

非阻塞代碼實例:

var fs = require("fs");

fs.readFile('input.txt', function (err, data) {
    if (err) return console.error(err);
    console.log(data.toString());
});

console.log("程序執行結束!");

結果:

程序執行結束! 菜鳥教程官網地址:www.runoob.com

結論:

在 Node 應用程序中,執行異步操做的函數將回調函數做爲最後一個參數, 回調函數接收錯誤對象做爲第一個參數。

以上兩個實例咱們瞭解了阻塞與非阻塞調用的不一樣。第一個實例在文件讀取完後才執行完程序。 第二個實例咱們不須要等待文件讀取完,這樣就能夠在讀取文件時同時執行接下來的代碼,大大提升了程序的性能。

所以,阻塞是按順序執行的,而非阻塞是不須要按順序的,因此若是須要處理回調函數的參數,咱們就須要寫在回調函數內。

 

node.js事件循環

Node.js 是單進程單線程應用程序,可是經過事件和回調支持併發,因此性能很是高。

Node.js 的每個 API 都是異步的,並做爲一個獨立線程運行,使用異步函數調用,並處理併發。

Node.js 基本上全部的事件機制都是用設計模式中觀察者模式實現。

Node.js 單線程相似進入一個while(true)的事件循環,直到沒有事件觀察者退出,每一個異步事件都生成一個事件觀察者,若是有事件發生就調用該回調函數.

事件驅動程序

Node.js 使用事件驅動模型,當web server接收到請求,就把它關閉而後進行處理,而後去服務下一個web請求。

當這個請求完成,它被放回處理隊列,當到達隊列開頭,這個結果被返回給用戶。

這個模型很是高效可擴展性很是強,由於webserver一直接受請求而不等待任何讀寫操做。(這也被稱之爲非阻塞式IO或者事件驅動IO)

在事件驅動模型中,會生成一個主循環來監聽事件,當檢測到事件時觸發回調函數。

 

整個事件驅動的流程就是這麼實現的,很是簡潔。有點相似於觀察者模式,事件至關於一個主題(Subject),而全部註冊到這個事件上的處理函數至關於觀察者(Observer)。

Node.js 有多個內置的事件,咱們能夠經過引入 events 模塊,並經過實例化 EventEmitter 類來綁定和監聽事件。

// 引入 events 模塊
var events = require('events');
// 建立 eventEmitter 對象
var eventEmitter = new events.EventEmitter();
// 綁定事件及事件的處理程序
eventEmitter.on('eventName', eventHandler);
// 觸發事件
eventEmitter.emit('eventName');

實例:

// 引入 events 模塊
var events = require('events');
// 建立 eventEmitter 對象
var eventEmitter = new events.EventEmitter();

// 建立事件處理程序
var connectHandler = function connected() {
   console.log('鏈接成功。');
  
   // 觸發 data_received 事件 
   eventEmitter.emit('data_received');
}

// 綁定 connection 事件處理程序
eventEmitter.on('connection', connectHandler);
 
// 使用匿名函數綁定 data_received 事件
eventEmitter.on('data_received', function(){
   console.log('數據接收成功。');
});

// 觸發 connection 事件 
eventEmitter.emit('connection');

console.log("程序執行完畢。");

執行代碼:

鏈接成功。 數據接收成功。 程序執行完畢。
相關文章
相關標籤/搜索