本文為您分享「Node.js 入門你需要知道的 10 個(gè)問題」這些問題可能也是面試中會(huì)被問到的,當(dāng)然問題不僅僅是這 10 道,因此,最近開源了一個(gè)新項(xiàng)目 Nodejs-Interview-Questions 專注于 Node.js 面試題的分享,提供了中英文版本,您也可以在線預(yù)覽: https://interview.nodejs.red/
Q1: 什么是 Node.js?
Node.js 是一個(gè)基于 Chrome V8 引擎的 JavaScript 運(yùn)行環(huán)境。它是一個(gè)開源和跨平臺(tái)的服務(wù)端應(yīng)用程序。任何人都可以編寫 JavaScript 代碼來開發(fā) Node.js 應(yīng)用程序。它可以運(yùn)行于 Microsoft Windows、Linux、 或 OS 系統(tǒng)。
Node.js 不是一個(gè)新的語言,也不僅僅是一個(gè)基于 JavaScript 的框架,它基于 Chrome 的 JavaScript 運(yùn)行時(shí),因此代碼的編寫和執(zhí)行與瀏覽器非常相似。
Node.js 功能
以下是 Node.js 的一些重要功能
- 高度可擴(kuò)展
Node.js 使用的單線程模型且采用了事件循環(huán)架構(gòu),使得編寫可擴(kuò)展性高的服務(wù)器變得既容易又安全。一些傳統(tǒng)的服務(wù)端語言會(huì)創(chuàng)建多線程來處理請(qǐng)求,通常創(chuàng)建線程都是有系統(tǒng)資源開銷的,因此也會(huì)有一些限制,而 Node.js 只創(chuàng)建一個(gè)線程來處理更多的請(qǐng)求。
- 事件驅(qū)動(dòng)和異步
Node.js 的所有 API 都是異步的。這意味著下一個(gè)請(qǐng)求來臨時(shí)可以直接處理而不用等待上一次的請(qǐng)求結(jié)果先返回。
- No Buffering
Node.js 從不緩沖任何任何數(shù)據(jù),參見What is No-Buffering feature of Node.js
我們?cè)S多人可能會(huì)對(duì) Node.js 感到困惑。它不是像 Apache 這樣的 Web 服務(wù)器。Node.js 提供了一種新方法來執(zhí)行我們的代碼。它是 JavaScript 的運(yùn)行時(shí)。Node.js 提供了創(chuàng)建 HTTP 服務(wù)器的方法,我們可以在這之上托管我們的應(yīng)用程序。
Source: Introduction To Node.js
Q2: 如何安裝 Node.js?
我們可以從 Node.js 官方網(wǎng)站 https://nodejs.org/en/ 下載安裝軟件。
nvm 安裝
這里推薦您使用 nvm 工具安裝,方便后期的升級(jí)進(jìn)行 Node.js 版本管理,以下為安裝步驟:
- 安裝 nvm:wget -qO- https://raw.githubusercontent.com/creationix/nvm/v0.33.2/install.sh | bash
- 查看所有 Node.js 版本:nvm ls-remote
- 查看本地 Node.js 版本:nvm ls
- 安裝 Node.js:nvm install v6.9.5
- 設(shè)置系統(tǒng)的默認(rèn) Node.js 版本:nvm alias default v6.9.5
驗(yàn)證安裝結(jié)果
在 Node.js 安裝成功之后,我們可以檢查它是否正常工作。
打開命令終端,輸入以下命令
$ node
之后將出現(xiàn) Node 提示符,我們寫入以下命令,運(yùn)行查看
console.log("hello world!");
按 Enter 鍵
Q3: 如何用 Node.js 監(jiān)聽 80 端口?
這是有陷阱的,在類似一些 Unix 系統(tǒng)中你不應(yīng)該嘗試監(jiān)聽 80 端口,這么做你需要擁有超級(jí)用戶權(quán)限,因此,不推薦你這么做。
盡管如此,如果你一定要讓應(yīng)用監(jiān)聽 80 端口,可以使用 Nginx 來實(shí)現(xiàn),在應(yīng)用前方加上一層反向代理。還是建議你監(jiān)聽大于 1024 的端口。
Q4: 什么是錯(cuò)誤優(yōu)先的回調(diào)函數(shù)?
錯(cuò)誤優(yōu)先回調(diào)函數(shù)用于同時(shí)返回錯(cuò)誤(error)和數(shù)據(jù)信息(data),返回值的第一個(gè)參數(shù)做為錯(cuò)誤信息描述,并且驗(yàn)證它是否出錯(cuò)(非錯(cuò) error 為 null),其它參數(shù)用于返回?cái)?shù)據(jù)。
fs.readFile(filePath, function(err, data) {
if (err) {
// 錯(cuò)誤信息處理
return console.log(err)
}
// return the data object
return data;
})
Q5: 你可以在 Node.js 中創(chuàng)建 Http 服務(wù)嗎?通過代碼來展示
在 Node.js 中創(chuàng)建一個(gè) Http 服務(wù)是很簡單的一件事情,我們可以通過 HTTP 模塊來完成這些操作。
const http = require('http');
const server = http.createServer((request, response) => {
if (request.url === '/hello') {
response.writeHead(200, {
'Content-Type': 'text/plain',
});
response.end('hello world!');
} else {
response.end('OK!');
}
});
server.listen(3000, '127.0.0.1', () => {
console.log('service is listening at http://127.0.0.1:3000');
});
Q6: Node.js 的核心組件有哪些?
Node.js 的核心組建是系統(tǒng) API、V8 引擎和 Libuv。
Libuv 庫
libuv 庫是一個(gè)跨平臺(tái)的支持事件驅(qū)動(dòng)的 I/O 庫。它是使用 C 和 C++ 語言為 Node.js 所開發(fā)的。但是它也被應(yīng)用于 Mozilla's 的 Rust、Luvit、Julia、pyuv 等其它的語言。
libuv 庫是 I/O 操作的核心部分,例如讀取文件和 OS 交互。
關(guān)于 Libuv 的學(xué)習(xí),可以參考 libuv中文教程
V8 引擎
來自于谷歌:“V8 是谷歌開源的高性能 JavaScript 引擎”,使用 C++ 開發(fā),并在谷歌瀏覽器中使用。V8 中實(shí)現(xiàn)的 ECMAScript 中指定 ECMA - 262 ,第 3版運(yùn)行在 Windows XP 和 Vista、Mac OS X 的 10.5 和 Linux 系統(tǒng)使用 IA - 32 或 ARM/MIPS 處理器。V8 可以獨(dú)立運(yùn)行,也可以嵌入到任何 C++ 應(yīng)用程序。
如果你感興趣想學(xué)習(xí)更多的 V8 引擎,請(qǐng)?jiān)L問 What is V8?
APIs (NodeJS Core Libs)
Node.js APIs 是根據(jù)您的請(qǐng)求去調(diào)用一些函數(shù)執(zhí)行一些業(yè)務(wù)操作。默認(rèn)情況下 Node.js 的 APIs 都是異步的,但是你想同步使用也是可以的(同步方式是不推薦的)。
例如,這個(gè) fs 模塊可以使用同步方式也可以使用異步方式。
var fs = require('fs');
fs.readFile('/files/help.txt', function(err, buf) {
// use fs.readFileSync() for sync operation. console.log(buf.toString());
});
Source: Introduction to NodeJS, A SSJS: Part I - Components Explained
Q7: 什么是“回調(diào)地獄”及如何避免它?
“回調(diào)地獄”是指嚴(yán)重的回調(diào)嵌套,這些回調(diào)嵌套使得代碼變得難以閱讀和維護(hù)。
以下是回調(diào)嵌套的示例:
query("SELECT clientId FROM clients WHERE clientName='picanteverde';", function(id){
query(`SELECT * FROM transactions WHERE clientId=${id}`, function(transactions){
transactions.each((transac) => {
query(`UPDATE transactions SET value = ${transac.value*0.1} WHERE id=${transac.id}`, (error) => {
if(!error){
console.log("success!!");
}else{
console.log("error");
}
});
});
});
});
在某種程度上,修復(fù)“回調(diào)地獄”的方式是模塊化。回調(diào)被分解為獨(dú)立的函數(shù),這些函數(shù)可以通過參數(shù)進(jìn)行傳遞。所以,針對(duì)以上代碼的第一個(gè)改進(jìn)如下所示:
const logError = (error) => {
if(!error){
console.log("success!!");
}else{
console.log("error");
}
},
updateTransaction = (t) => {
query(`UPDATE transactions SET value = ${t.value*0.1} WHERE id=${t.id}`, logError);
},
handleTransactions = (transactions) => {
transactions.each(updateTransaction);
},
handleClient = (id) => {
query(`SELECT * FROM transactions WHERE clientId=${id}`, handleTransactions);
};
query("SELECT clientId FROM clients WHERE clientName='picanteverde';",handleClient);
盡管這個(gè)代碼相比第一個(gè)示例更容易易讀,而且我們創(chuàng)建的的函數(shù)還可以得到復(fù)用。但是在某些情況下,我們想要使程序更健壯可通過 Promise 來解決。
此外,generators 也提供了強(qiáng)大的回調(diào)地獄解決方案,使用它可以解決不同回調(diào)之間的依賴關(guān)系。然而 generators 會(huì)更高級(jí)一些使用起來會(huì)復(fù)雜一些。關(guān)于 Generators 更多信息可以閱讀這篇文章 Generators in Node.js
然而,以上的雖然能很好解決回調(diào)地獄問題,但是目前有了更好的方案 Async/Await。使用 Async/Await 需要注意 Node.js 版本要在 v7.5 版本之上。
Source: 8 Essential Node.js Interview Questions
Q8: 什么是 Node.js 的事件驅(qū)動(dòng)編程?
事件驅(qū)動(dòng)程序是由事件(click、load 等)決定的代碼流程術(shù)語。它是當(dāng)今流行編程語言(例如 C#、Java)里一個(gè)最基本的里程碑,在這里不會(huì)詳細(xì)講述。在 Node.js 中或者一些其它類型的 JavaScript 項(xiàng)目中,我們都在使用事件驅(qū)動(dòng)編程。也許你并不知道事件驅(qū)動(dòng)編程,但是在一些頁面加載或按鈕單擊事件中,你已經(jīng)在使用了。
舉一個(gè)典型的事件驅(qū)動(dòng)流程的例子,看下它是如何在 Node.js 中完成中:
result = getJSONfromDestination();
binddata(result);
上述操作是一個(gè)阻塞 I/O(單線程模式下將會(huì)等待這個(gè)阻塞 I/O 完成之后才會(huì)進(jìn)行下一步)
現(xiàn)在讓我們看看異步方式該如何進(jìn)行(非阻塞 I/O 進(jìn)程)
json_finished = function(result){
binddata(result);
}
getJSONfromDestination(jsonfinished);
如上所示,這是一個(gè)非阻塞的例子,因?yàn)?json_finished 不是你所想向的那樣會(huì)直接工作。當(dāng)您調(diào)用 getJSONfromDestination 函數(shù)并將 jsonfinished 做為參數(shù)傳遞時(shí),它才開始工作。
Source: NodeJS Series #6: Event - Driven Programming
Q9: 什么是 NPM? 在 Node.js 中什么時(shí)候需要 NPM?
NPM 是 Node.js 中的包管理器。允許我們?yōu)?Node.js 安裝各種模塊,這個(gè)包管理器為我們提供了安裝、刪除等其它命令來管理模塊。這里有一點(diǎn)我們需要注意,我們必須要有一個(gè) package.json 文件或 node_modules 目錄安裝模塊到本地。
NPM 最好的一點(diǎn)是它會(huì)在本地存儲(chǔ)我們所安裝的依賴項(xiàng),存在于 package.json 的 dependencies 對(duì)象里。例如,如果一個(gè)模塊 X 使用了模塊 A 版本為 1.0,模塊 Y 使用了模塊 A 版本為 1.5,那么模塊 X 或 Y 都將在本地?fù)碛凶约簩?duì)應(yīng)的模塊 A 的副本。
// 模塊 X
{
"name": "X",
"dependencies": {
"A": "^1.0"
}
}
// 模塊 Y
{
"name": "Y",
"dependencies": {
"A": "^1.5"
}
}
需要 NPM 包
當(dāng)我們?cè)陂_發(fā)一些 Node.js 項(xiàng)目時(shí),可能會(huì)遇到一些地方需要 NPM,例如鏈接 Redis、MongoDB 或者發(fā)送請(qǐng)求 Request 等,有了這些模塊可以使我們更專注于業(yè)務(wù)開發(fā),當(dāng)然有時(shí)你會(huì)有些特別的需求,這時(shí)可能需要自己去封裝一個(gè) NPM 模塊,實(shí)現(xiàn)復(fù)用。
點(diǎn)擊下面 Source 閱讀更多關(guān)于 NPM 的相關(guān)內(nèi)容
Source: How to Create Nodejs Module and Publish Over to Npm
Q10: Node.js 可以做什么? 10 個(gè) Node.js 的應(yīng)用場景?
Node.js 可以做 Web 服務(wù)端、命令行工具 (Java, PHP 可以做的 JS 也可以做),現(xiàn)在讓我們看下 Node.js 的 10 個(gè)應(yīng)用場景:
- Web 開發(fā): Express + EJS + MongoDB(mongoose)/Mysql
- REST 開發(fā): Restify
- IM 即時(shí)聊天: Express + Socket.io
- 網(wǎng)絡(luò)爬蟲: Cheerio/request
- 博客系統(tǒng): Hexo
- 網(wǎng)絡(luò)論壇: Nodeclub
- Web 幻燈片: Cleaver
- 前端構(gòu)建工具: bower.js
- OAuth 認(rèn)證: Passport
- 定時(shí)任務(wù)工具: Later
Source: What does node. js do? 10 application scenarios for node. js
本片文章首發(fā)于慕課網(wǎng):https://www.imooc.com/article/289202