Node.js 應(yīng)用跑得更快 10 個(gè)技巧
來源:易賢網(wǎng) 閱讀:694 次 日期:2016-07-08 15:46:29
溫馨提示:易賢網(wǎng)小編為您整理了“Node.js 應(yīng)用跑得更快 10 個(gè)技巧”,方便廣大網(wǎng)友查閱!

NodeJS是一個(gè)服務(wù)器端JavaScript解釋器,它將改變服務(wù)器應(yīng)該如何工作的概念。它的目標(biāo)是幫助程序員構(gòu)建高度可伸縮的應(yīng)用程序,編寫能夠處理數(shù)萬條同時(shí)連接到一個(gè)(只有一個(gè))物理機(jī)的連接代碼。

 Node.js 受益于它的事件驅(qū)動和異步的特征,已經(jīng)很快了。但是,在現(xiàn)代網(wǎng)絡(luò)中只是快是不行的。如果你打算用 Node.js 開發(fā)你的下一個(gè)Web 應(yīng)用的話,那么你就應(yīng)該無所不用其極,讓你的應(yīng)用更快,異常的快。本文將介紹 10 條,經(jīng)過檢驗(yàn)得知可大大提高 Node 應(yīng)用的技巧。廢話不多說,讓我們逐條來看看。

1.并行

創(chuàng)建 Web 應(yīng)用的時(shí)候,你可能要多次調(diào)用內(nèi)部 API 來獲取各種數(shù)據(jù)。比如說,假設(shè)在 Dashboard 頁面上,你要執(zhí)行下面這幾個(gè)調(diào)用:

用戶信息 -getUserProfile().

當(dāng)前活動 -getRecentActivity().

訂閱內(nèi)容 -getSubscriptions().

通知內(nèi)容 -getNotifications().

為了拿到這些信息,你應(yīng)該會為每個(gè)方法創(chuàng)建獨(dú)立的中間件,然后將它們鏈接到 Dashboard 路由上。不過問題是,這些方法的執(zhí)行是線性的,上一個(gè)沒結(jié)束之前下一個(gè)不會開始??尚薪鉀Q案是并行調(diào)用它們。

如你所知由于異步性,Node.js 非常擅長并行調(diào)用多個(gè)方法。我們不能暴殄天物。我上面提到的那些方法沒有依賴性,所以我們可以并行執(zhí)行它們。這樣我們可以削減中間件數(shù)量,大幅提高速度。

我們可以用 async.js 來處理并行,它是一個(gè)專門用來調(diào)教 JavaScript 異步的 Node 模塊。下面代碼演示怎樣用 async.js 并行調(diào)用多個(gè)方法的:

function runInParallel() {

async.parallel([

getUserProfile,

getRecentActivity,

getSubscriptions,

getNotifications

], function(err, results) {

//This callback runs when all the functions complete

});

}

如果你想更深入了解 async.js ,請移步它的 GitHub 頁面。

2.異步

根據(jù)設(shè)計(jì) Node.js 是單線程的?;谶@點(diǎn),同步代碼會堵塞整個(gè)應(yīng)用。比如說,多數(shù)的文件系統(tǒng) API 都有它們的同步版本。下面代碼演示了文件讀取的同步和異步兩種操作:

// Asynchronous

fs.readFile('file.txt', function(err, buffer) {

var content = buffer.toString();

});

// Synchronous

var content = fs.readFileSync('file.txt').toString();

不過要是你執(zhí)行那種長時(shí)間的阻塞操作,主線程就會被阻塞到這些操作完成為止。這大大降低你應(yīng)用的性能。所以,最好確保你的代碼里用的都是異步版本 API,最起碼你應(yīng)該在性能節(jié)點(diǎn)異步。而且,你在選用第三方模塊的時(shí)候也要很小心。因?yàn)楫?dāng)你想方設(shè)法把同步操作從你代碼中剔除之后,一個(gè)外部庫的同步調(diào)用會讓你前功盡棄,降低你的應(yīng)用性能

3.緩存

如果你用到一些不經(jīng)常變化的數(shù)據(jù),你應(yīng)該把它們緩存起來,改善性能。比如說,下面的代碼是獲取最新帖子并顯示的例子:

var router = express.Router();

router.route('/latestPosts').get(function(req, res) {

Post.getLatest(function(err, posts) {

if (err) {

throw err;

}

res.render('posts', { posts: posts });

});

});

如果你不經(jīng)常發(fā)貼的話,你可以把帖子列表緩存起來,然后一段時(shí)間之后再把它們清理掉。比如,我們可以用 Redis 模塊來達(dá)到這個(gè)目的。當(dāng)然,你必須在你的服務(wù)器上裝 Redis。然后你可以用叫做 node_redis 的客戶端來保存鍵/值對。下面的例子演示我們怎么緩存帖子:

var redis = require('redis'),

client = redis.createClient(null, null, { detect_buffers: true }),

router = express.Router();

router.route('/latestPosts').get(function(req,res){

client.get('posts', function (err, posts) {

if (posts) {

return res.render('posts', { posts: JSON.parse(posts) });

}

Post.getLatest(function(err, posts) {

if (err) {

throw err;

}

client.set('posts', JSON.stringify(posts)); 

res.render('posts', { posts: posts });

});

});

});

看到了吧,我們首先檢查 Redis 緩存,看看是否有帖子。如果有,我們從緩存中拿這些帖子列表。否則我們就檢索數(shù)據(jù)庫內(nèi)容,然后把結(jié)果緩存。此外,一定時(shí)間之后,我們可以清理 Redis 緩存,這樣就可以更新內(nèi)容了。

4.gzip壓縮

開啟 gzip 壓縮對你的 Web 應(yīng)用會產(chǎn)生巨大影響。當(dāng)一個(gè) gzip 壓縮瀏覽器請求某些資源的時(shí)候,服務(wù)器會在響應(yīng)返回給瀏覽器之前進(jìn)行壓縮。如果你不用 gzip 壓縮你的靜態(tài)資源,瀏覽器拿到它們可能會花費(fèi)更長時(shí)間。

在 Express 應(yīng)用中,我們可以用內(nèi)建 express.static() 中間件來處理靜態(tài)內(nèi)容。此外,還可以用 compression 中間件壓縮和處理靜態(tài)內(nèi)容。下面是使用例:

var compression = require('compression');

app.use(compression()); //use compression 

app.use(express.static(path.join(__dirname, 'public')));

5.盡量在客戶端渲染

現(xiàn)在有超多功能強(qiáng)勁的客戶端 MVC/MVVM 框架,比如說 AngularJS, Ember, Meteor, 等等,構(gòu)建一個(gè)單頁面應(yīng)用變得非常簡單?;旧希阒灰_一個(gè) API,返回 JSON 響應(yīng)給客戶端就可以了,而不需要在服務(wù)端渲染頁面。

在客戶端,你可以用框架來組織 JSON 然后把它們顯示在 UI 上。服務(wù)端只發(fā)送 JSON 響應(yīng)可以節(jié)省帶寬,改善性能,因?yàn)槟悴恍枰诿總€(gè)響應(yīng)里面都返回布局標(biāo)記了,對吧,你只需要返回純 JSON,然后在客戶端渲染它們。

6.不要在Session存儲太多數(shù)據(jù)

典型的 Express 頁面應(yīng)用, Session 數(shù)據(jù)默認(rèn)是保存在內(nèi)存中的。當(dāng)你把太多數(shù)據(jù)保存在 Session 的時(shí)候,會導(dǎo)致服務(wù)器開銷顯著增大。所以,要么你切換到別的儲存方式來保存 Session 數(shù)據(jù),要么盡量減少存儲在 Session 中的數(shù)據(jù)量。

比如說,當(dāng)用戶登錄到你的應(yīng)用的時(shí)候,你可以只在 Session 中保存他們的 ID 而不是整個(gè)用戶數(shù)據(jù)對象。還有,對于那些你能夠從 id 拿到對象的查詢,你應(yīng)該會喜歡用 MongoDB 或者 Redis 來存儲 session 數(shù)據(jù)。

7.優(yōu)化查詢

假設(shè)你有個(gè)博客,你要在主頁上顯示最新帖子。你可能會通過 Mongoose 這樣取數(shù)據(jù):

Post.find().limit(10).exec(function(err, posts) {

//send posts to client

});

不過問題是 Mongoose 的 find() 方法會把對象的所有字段都查詢出來,而許多字段在主頁上并不要求。比如說,commentsis 保存的是特定帖子的回復(fù)。我們不需要顯示文章回復(fù),所以我們可以在查詢的時(shí)候把它給剔除掉。這無疑會提高速度??梢韵襁@樣優(yōu)化上面那條查詢:

Post.find().limit(10).exclude('comments').exec(function(err, posts) {

//send posts to client

});

8.用標(biāo)準(zhǔn)的V8方法

集合上的一些操作,比如 map,reduce,和 forEach 不一定支持所有瀏覽器。我們可以通過前臺的庫解決部分瀏覽器兼容性問題。但對于 Node.js,你要確切知道 Google 的 V8 JavaScript 引擎支持哪些操作。這樣,你就可以在服務(wù)端直接用這些內(nèi)建方法來操作集合了。

9.在 Node 前面用 Nginx

Nginx 是個(gè)微小型輕量 Web 服務(wù)器,用它可以降低你的 Node.js 服務(wù)器的負(fù)載。你可以把靜態(tài)資源配置到 nginx 上,而不是在 Node 上。你可以在 nginx 上用 gzip 壓縮響應(yīng),讓所有的響應(yīng)都變得更小。所以,如果你有個(gè)正在營運(yùn)的產(chǎn)品,我覺得你應(yīng)該會想用 nginx 來改善運(yùn)行速度的。

10.打包JavaScript

最后,你還可以大大提高頁面應(yīng)用速度,通過把多個(gè) JS 文件打包。當(dāng)瀏覽器在頁面渲染中碰到 <script\> 元素的時(shí)候會被堵塞,直到拿到這個(gè)腳本才繼續(xù)運(yùn)行(除非設(shè)置了異步屬性)。比如,如果你的頁面有五個(gè) JavaScript 文件,瀏覽器會發(fā)出五個(gè)獨(dú)立的 HTTP 請求來獲取他們。如果把這五個(gè)文件壓縮打包成一個(gè),整體性能將可以大幅提升。CSS 文件也是一樣。你可以用諸如 Grunt/Gulp 這樣的編譯工具來打包你的資源文件。

通過以上十個(gè)方面給大家介紹了Node.js 應(yīng)用跑得更快的技巧,希望對大家有所幫助!

更多信息請查看網(wǎng)絡(luò)編程
易賢網(wǎng)手機(jī)網(wǎng)站地址:Node.js 應(yīng)用跑得更快 10 個(gè)技巧
由于各方面情況的不斷調(diào)整與變化,易賢網(wǎng)提供的所有考試信息和咨詢回復(fù)僅供參考,敬請考生以權(quán)威部門公布的正式信息和咨詢?yōu)闇?zhǔn)!

2025國考·省考課程試聽報(bào)名

  • 報(bào)班類型
  • 姓名
  • 手機(jī)號
  • 驗(yàn)證碼
關(guān)于我們 | 聯(lián)系我們 | 人才招聘 | 網(wǎng)站聲明 | 網(wǎng)站幫助 | 非正式的簡要咨詢 | 簡要咨詢須知 | 新媒體/短視頻平臺 | 手機(jī)站點(diǎn) | 投訴建議
工業(yè)和信息化部備案號:滇ICP備2023014141號-1 云南省教育廳備案號:云教ICP備0901021 滇公網(wǎng)安備53010202001879號 人力資源服務(wù)許可證:(云)人服證字(2023)第0102001523號
云南網(wǎng)警備案專用圖標(biāo)
聯(lián)系電話:0871-65099533/13759567129 獲取招聘考試信息及咨詢關(guān)注公眾號:hfpxwx
咨詢QQ:1093837350(9:00—18:00)版權(quán)所有:易賢網(wǎng)
云南網(wǎng)警報(bào)警專用圖標(biāo)