新聞中心
臨陣磨槍,不快也光。對于成功的編程面試來說,準備和知識面一樣重要。準備使你有信心參加面試,而不用擔心莫名的緊張情緒。如果第一次參加編程面試,這一點尤其重要。

創(chuàng)新互聯(lián)建站服務項目包括汾陽網(wǎng)站建設、汾陽網(wǎng)站制作、汾陽網(wǎng)頁制作以及汾陽網(wǎng)絡營銷策劃等。多年來,我們專注于互聯(lián)網(wǎng)行業(yè),利用自身積累的技術優(yōu)勢、行業(yè)經(jīng)驗、深度合作伙伴關系等,向廣大中小型企業(yè)、政府機構等提供互聯(lián)網(wǎng)行業(yè)的解決方案,汾陽網(wǎng)站推廣取得了明顯的社會效益與經(jīng)濟效益。目前,我們服務的客戶以成都為中心已經(jīng)輻射到汾陽省份的部分城市,未來相信會繼續(xù)擴大服務區(qū)域并繼續(xù)獲得客戶的支持與信任!
為幫助 Node.js 開發(fā)人員更好的面試,我列出了 15 個常見的 Node.js 和網(wǎng)絡開發(fā)相關的面試問題。
在本文中,我們將重點討論 Node.js 相關問題。但是,請記住 JavaScript 問題在 Node.js 面試中也經(jīng)常問到,所以準備一些對你來說沒什么壞處。不久前我們寫了一篇關于常見 JavaScript 面試問題的帖子,涵蓋了所有這些基礎。
https://livecodestream.dev/post/20-common-javascript-interview-questions/
現(xiàn)在,讓我們深入了解面試中可能會問到的 Node.js 問題。
1. Node.js 與 JavaScript 有什么不同?
2. 什么時候用 Node.js?
Node.js 是異步的、事件驅(qū)動的、非阻塞的和單線程的,使得它成為開發(fā)下面應用程序的完美候選:
- 實時應用程序,如聊天和提供實時更新的應用程序
- 將視頻或其他多媒體內(nèi)容流式傳輸給大量觀眾的流式應用程序
- 其他 I/O 密集型應用程序,如協(xié)作平臺
- 遵循微服務架構的網(wǎng)絡后端
然而,Node.js 的特性使得它對于其他類型的應用程序來說不是一個理想的選擇。執(zhí)行 CPU 密集型任務的應用程序(如復雜的數(shù)學計算)在使用 CPU 時表現(xiàn)不佳,因為 Node.js 是單線程的。
如果你想了解更多這方面的信息,請查看我們的文章 Node.js 架構以及何時在項目中使用。
3. EventEmitter 做了什么?
Node.js 中任何對象發(fā)出的事件都是 EventEmitter 類的實例,就像 http 模塊。
所有 EventEmitter 類都可以使用 eventEmitter.on() 函數(shù)將事件偵聽器附加到事件。然后一旦捕捉到這樣的事件,就會同步地逐個調(diào)用它的偵聽器。
- const events = require("events");
- const eventEmitter = new events.EventEmitter();
- const eventListener = function(){
- console.log("event triggered");
- }
- eventEmitter.on("emitted", eventListener);
- eventEmitter.emit("emitted");
4. 事件循環(huán)是什么?
單線程的 Node.js 必須是非阻塞的,以防止線程阻塞在需要很長時間才能完成的任務上,事件循環(huán)負責實現(xiàn)這種非阻塞行為,它使用應用程序線程調(diào)度掛起的任務。
Node.js 在任務完成時通過回調(diào)來處理異步函數(shù)返回的響應。與創(chuàng)建任務的事件類似,任務完成后也會發(fā)出一個事件。Node.js 將需要處理的事件添加到事件隊列。
事件循環(huán)對事件隊列中的事件進行迭代,并安排何時執(zhí)行其關聯(lián)的回調(diào)函數(shù)。
5. 流是什么?
Stream 流是從源讀取或?qū)懭霐?shù)據(jù)并將其傳輸?shù)竭B續(xù)流目標的管道。有四種類型:
- 可讀
- 可寫的
- 可讀寫
- 先寫入,再讀出來
每個流也是一個 EventEmitter。這意味著流對象可以在流上沒有數(shù)據(jù)、流上有可用數(shù)據(jù)或流中的數(shù)據(jù)在程序刷新時發(fā)出事件。
- const fs = require("fs");
- const readableStream = fs.createReadStream("test.txt");
- let content = "";
- readableStream.on("data", (chunk) => {
- content += chunk;
- });
- readableStream.on("end", () => {
- console.log(content);
- });
6. readFile 和 createReadStream 函數(shù)有什么區(qū)別?
readFile 函數(shù)異步讀取文件的全部內(nèi)容,并存儲在內(nèi)存中,然后再傳遞給用戶。
createReadStream 使用一個可讀的流,逐塊讀取文件,而不是全部存儲在內(nèi)存中。
與 readFile 相比,createReadStream 使用更少的內(nèi)存和更快的速度來優(yōu)化文件讀取操作。如果文件相當大,用戶不必等待很長時間直到讀取整個內(nèi)容,因為讀取時會先向用戶發(fā)送小塊內(nèi)容。
- const fs = require("fs");
- fs.readFile("test.txt", (err, content) => {
- console.log(content);
- });
7. 如何處理 Node.js 中未捕獲的異常?
我們可以在進程級別捕獲應用程序中未捕獲的異常。為此將偵聽器附加到 process 全局對象:
- process.on("uncaughtException", (err) => {
- console.log("exception caught: ", err);
- });
8. Node.js 能否充分利用多核處理器?
(默認的)Node.js 應用程序總是單線程的,即使在多核處理器上運行,應用程序也能只使用一個處理器。
但是 Node.js 的核心模塊之一 Cluster 支持 Node.js 應用程序開啟多核,允許我們創(chuàng)建多個工作進程,這些進程可以在多個內(nèi)核上并行運行,并共享一個端口來偵聽事件。
每個進程使用 IPC 與主線程通信,并根據(jù)需要將服務器句柄傳遞給其他進程。主進程可以偵聽端口本身并以循環(huán)方式將每個新連接傳遞給子進程,也可以將端口分配給子進程以便子進程偵聽請求。
9. 反應堆設計模式是什么?
反應堆設計模式是,Node.js 將回調(diào)函數(shù)(處理程序)附加到每個 I/O 操作,然后創(chuàng)建請求時將處理程序提交給解復用器。
解復用器收集應用程序中發(fā)出的每個 I/O 請求,并將它們作為隊列中的事件進行排隊。這個隊列就是我們所說的事件隊列。將事件排隊后,解復用器返回應用程序線程的控制。
同時,事件循環(huán)遍歷事件隊列中的每個事件,并調(diào)用附加的回調(diào)來處理事件響應。
這就是 Node.js 中所使用的反應堆模式。
10. 單線程與多線程網(wǎng)絡后端相比有哪些好處?
盡管 Node.js 是單線程的,但是大多數(shù)用于后端開發(fā)的編程語言都提供多線程來處理應用程序操作。
為什么單線程有利于后端開發(fā)?
- 開發(fā)人員更容易實現(xiàn)應用程序。我們的應用程序在生產(chǎn)過程中不會突然遇到意外的競爭條件。
- 單線程應用程序易于擴展。
- 它們可以毫不延遲地在一個時刻收到的大量用戶請求提供服務。相比之下,當流量較大時,多線程后端必須等待線程池中的線程釋放,才能為用戶請求提供服務。利用 Node.js 的非阻塞特性,用戶請求不會在單個線程上掛起太長時間(只有在操作不是 CPU 密集型時)。
11. REPL 是什么?
REPL 代表 Read Eval Print Loop,是一個虛擬環(huán)境,可以在其中輕松地運行編程語言。Node.js 帶有一個內(nèi)置的 REPL 來運行 JavaScript 代碼,類似于我們在瀏覽器中用來運行 JavaScript 代碼的控制臺。
要啟動 Node.js REPL,只需在命令行上運行 node,然后寫一行 JavaScript 代碼,就可以在下一行看到它的輸出。
12. process.nextTick 和 setImmediate 有什么區(qū)別?
傳遞給 setImmediate 函數(shù)的回調(diào)將在事件隊列上的下一次迭代中執(zhí)行。
另一方面,回調(diào)傳遞給 process.nextTick 在下一次迭代之前以及程序中當前運行的操作完成之后執(zhí)行。在應用程序啟動時,開始遍歷事件隊列之前調(diào)用它的回調(diào)。
因此,回調(diào) process.nextTick 總是在 setImmediate 之前調(diào)用。
下面代碼段:
- setImmediate(() => {
- console.log("first");
- })
- process.nextTick(() => {
- console.log("second");
- })
- console.log("third");
將按順序輸出:
- third
- second
- first
13. stub 是什么?
測試應用程序時使用 stub,模擬給定組件或模塊的行為,你可以將精力集中在要測試的代碼部分。通過使用 stub 代替與測試無關的組件,不必擔心外部組件會影響結(jié)果。
例如,如果正在測試的組件在預期測試的部分之前有一個文件讀取操作,則可以使用 stub 來模擬該行為并返回模擬內(nèi)容,而不用實際讀取文件。
在 Node.js 中,我們使用像 Sinon 這樣的庫來實現(xiàn)(譯者注,Sinon 在測試中替換某部分代碼,減少測試項編寫的復雜度 https://sinonjs.org)。
14. 為什么在 express 中分離“應用程序”和“服務器”是一種好的做法?
通過在 Express 中分離應用程序和服務器,可以將 API 實現(xiàn)與網(wǎng)絡相關配置分開。在不執(zhí)行網(wǎng)絡調(diào)用的情況下執(zhí)行 API 測試,保證了更快的測試執(zhí)行和更好的代碼覆蓋度量。
要實現(xiàn)這種分離,應該在單獨的文件中聲明 API 和 server,對應 app.js 和 server.js:
- // app.js
- const express = require("express");
- const app = express();
- app.use("/", index);
- app.use("/contact", contact);
- app.use("/user", user);
- module.exports = app;
- // server.js
- const http = require("http");
- const app = require("/app");
- app.set('port', process.env.PORT);
- const httphttp = http.createServer(app);
15. 什么是 yarn 和 npm?為什么要用 yarn 代替 npm 呢?
npm 是與 Node.js 自帶的默認包管理器,它有一個大型的公共庫和私有庫,存儲在 npm registry 的數(shù)據(jù)庫中(譯者注,官方默認中心庫 http://registry.npmjs.org/,國內(nèi)淘寶鏡像 http://registry.npm.taobao.org/),用戶可以通過 npm 命令行訪問該數(shù)據(jù)庫。在 npm 的幫助下,用戶可以輕松管理項目中的依賴項。
yarn 也是一個包管理器,為了解決 npm 的一些缺點。yarn 依賴 npm 注冊中心為用戶提供對包訪問。yarn 底層結(jié)構基于 npm,如果從 npm 遷移到 yarn,項目結(jié)構和工作流不需要大改。
就像之前提到的,在某些情況下,yarn 提供了比 npm 更好的功能。與 npm 不同的是,它會緩存下載的每個包,不必重新下載。
通過校驗和驗證包的完整性來提供更好的安全性,保證在某個系統(tǒng)上運行的包在任何其他系統(tǒng)中的工作方式完全相同,這就是為什么選擇 yarn 而不是 npm 來進行包管理。
結(jié)論
在本文中,討論了 15 個最常見的 Node.js 面試問題,幫助你為下一次面試做準備。知道你可能被問到的問題和答案,面試就不再緊張了。
祝你好運,面試成功!
網(wǎng)站題目:15個常見的Node.js面試問題及答案
本文URL:http://www.fisionsoft.com.cn/article/djeghpc.html


咨詢
建站咨詢
