Nodejs如何進(jìn)行性能監(jiān)控和分析優(yōu)化
Node.js應(yīng)用可能因?yàn)楦卟l(fā)、內(nèi)存泄漏、CPU密集型任務(wù)等原因?qū)е滦阅芟陆?,影響用戶體驗(yàn)甚至系統(tǒng)穩(wěn)定性。通過性能監(jiān)控和分析,我們可以及時(shí)發(fā)現(xiàn)潛在問題,并針對性地進(jìn)行優(yōu)化,確保系統(tǒng)正常運(yùn)行且具備良好的性能表現(xiàn)。
一、 Nodejs性能衡量指標(biāo)
Nodejs
作為一門服務(wù)端語言,性能方面尤為重要,其衡量指標(biāo)一般有如下:
- CPU
- 內(nèi)存
- I/O
- 網(wǎng)絡(luò)
CPU
主要分成了兩部分:
- CPU負(fù)載:在某個(gè)時(shí)間段內(nèi),占用以及等待CPU的進(jìn)程總數(shù)
- CPU使用率:CPU時(shí)間占用狀況,等于 1 - 空閑CPU時(shí)間(idle time) / CPU總時(shí)間
這兩個(gè)指標(biāo)都是用來評估系統(tǒng)當(dāng)前CPU的繁忙程度的量化指標(biāo)
Nodejs
應(yīng)用一般不會消耗很多的CPU
,如果CPU
占用率高,則表明應(yīng)用存在很多同步操作,導(dǎo)致異步任務(wù)回調(diào)被阻塞
內(nèi)存指標(biāo)
內(nèi)存是一個(gè)非常容易量化的指標(biāo)。 內(nèi)存占用率是評判一個(gè)系統(tǒng)的內(nèi)存瓶頸的常見指標(biāo)。 對于Node來說,內(nèi)部內(nèi)存堆棧的使用狀態(tài)也是一個(gè)可以量化的指標(biāo)
// /app/lib/memory.js const os = require('os'); // 獲取當(dāng)前Node內(nèi)存堆棧情況 const { rss, heapUsed, heapTotal } = process.memoryUsage(); // 獲取系統(tǒng)空閑內(nèi)存 const sysFree = os.freemem(); // 獲取系統(tǒng)總內(nèi)存 const sysTotal = os.totalmem(); module.exports = { memory: () => { return { sys: 1 - sysFree / sysTotal, // 系統(tǒng)內(nèi)存占用率 heap: heapUsed / headTotal, // Node堆內(nèi)存占用率 node: rss / sysTotal, // Node占用系統(tǒng)內(nèi)存的比例 } } }
- rss:表示node進(jìn)程占用的內(nèi)存總量。
- heapTotal:表示堆內(nèi)存的總量。
- heapUsed:實(shí)際堆內(nèi)存的使用量。
- external :外部程序的內(nèi)存使用量,包含Node核心的C++程序的內(nèi)存使用量
在Nodejs
中,一個(gè)進(jìn)程的最大內(nèi)存容量為1.5GB。因此我們需要減少內(nèi)存泄露
磁盤 I/O
硬盤的IO
開銷是非常昂貴的,硬盤 IO 花費(fèi)的 CPU 時(shí)鐘周期是內(nèi)存的 164000 倍
內(nèi)存 IO
比磁盤IO
快非常多,所以使用內(nèi)存緩存數(shù)據(jù)是有效的優(yōu)化方法。常用的工具如 redis
、memcached
等
并不是所有數(shù)據(jù)都需要緩存,訪問頻率高,生成代價(jià)比較高的才考慮是否緩存,也就是說影響你性能瓶頸的考慮去緩存,并且而且緩存還有緩存雪崩、緩存穿透等問題要解決
二、Nodejs如何監(jiān)控
關(guān)于性能方面的監(jiān)控,一般情況都需要借助工具來實(shí)現(xiàn)
這里采用Easy-Monitor 2.0
,其是輕量級的 Node.js
項(xiàng)目內(nèi)核性能監(jiān)控 + 分析工具,在默認(rèn)模式下,只需要在項(xiàng)目入口文件 require
一次,無需改動任何業(yè)務(wù)代碼即可開啟內(nèi)核級別的性能監(jiān)控分析
使用方法如下:
在你的項(xiàng)目入口文件中按照如下方式引入,當(dāng)然請傳入你的項(xiàng)目名稱:
const easyMonitor = require('easy-monitor'); easyMonitor('你的項(xiàng)目名稱');
打開你的瀏覽器,訪問 http://localhost:12333
,即可看到進(jìn)程界面
關(guān)于定制化開發(fā)、通用配置項(xiàng)以及如何動態(tài)更新配置項(xiàng)詳見官方文檔
三、Nodejs如何優(yōu)化
關(guān)于Nodejs
的性能優(yōu)化的方式有:
- 使用最新版本Node.js
- 正確使用流 Stream
- 代碼層面優(yōu)化
- 內(nèi)存管理優(yōu)化
使用最新版本Node.js
每個(gè)版本的性能提升主要來自于兩個(gè)方面:
- V8 的版本更新
- Node.js 內(nèi)部代碼的更新優(yōu)化
正確使用流 Stream
在Nodejs
中,很多對象都實(shí)現(xiàn)了流,對于一個(gè)大文件可以通過流的形式發(fā)送,不需要將其完全讀入內(nèi)存
const http = require('http'); const fs = require('fs'); // bad http.createServer(function (req, res) { fs.readFile(__dirname + '/data.txt', function (err, data) { res.end(data); }); }); // good http.createServer(function (req, res) { const stream = fs.createReadStream(__dirname + '/data.txt'); stream.pipe(res); });
代碼層面優(yōu)化
合并查詢,將多次查詢合并一次,減少數(shù)據(jù)庫的查詢次數(shù)
// bad for user_id in userIds let account = user_account.findOne(user_id) // good const user_account_map = {} // 注意這個(gè)對象將會消耗大量內(nèi)存。 user_account.find(user_id in user_ids).forEach(account){ user_account_map[account.user_id] = account } for user_id in userIds var account = user_account_map[user_id]
內(nèi)存管理優(yōu)化
在 V8 中,主要將內(nèi)存分為新生代和老生代兩代:
- 新生代:對象的存活時(shí)間較短。新生對象或只經(jīng)過一次垃圾回收的對象
- 老生代:對象存活時(shí)間較長。經(jīng)歷過一次或多次垃圾回收的對象
若新生代內(nèi)存空間不夠,直接分配到老生代
通過減少內(nèi)存占用,可以提高服務(wù)器的性能。如果有內(nèi)存泄露,也會導(dǎo)致大量的對象存儲到老生代中,服務(wù)器性能會大大降低
如下面情況:
const buffer = fs.readFileSync(__dirname + '/source/index.htm'); app.use( mount('/', async (ctx) => { ctx.status = 200; ctx.type = 'html'; ctx.body = buffer; leak.push(fs.readFileSync(__dirname + '/source/index.htm')); }) ); const leak = [];
leak
的內(nèi)存非常大,造成內(nèi)存泄露,應(yīng)當(dāng)避免這樣的操作,通過減少內(nèi)存使用,是提高服務(wù)性能的手段之一
而節(jié)省內(nèi)存最好的方式是使用池,其將頻用、可復(fù)用對象存儲起來,減少創(chuàng)建和銷毀操作
例如有個(gè)圖片請求接口,每次請求,都需要用到類。若每次都需要重新new這些類,并不是很合適,在大量請求時(shí),頻繁創(chuàng)建和銷毀這些類,造成內(nèi)存抖動
使用對象池的機(jī)制,對這種頻繁需要創(chuàng)建和銷毀的對象保存在一個(gè)對象池中。每次用到該對象時(shí),就取對象池空閑的對象,并對它進(jìn)行初始化操作,從而提高框架的性能
到此這篇關(guān)于Nodejs如何進(jìn)行性能監(jiān)控和分析優(yōu)化的文章就介紹到這了,更多相關(guān)Nodejs性能優(yōu)化內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Node.js使用officecrypto-tool實(shí)現(xiàn)讀取加密的Excel和Word文檔
這篇文章主要為大家詳細(xì)介紹了Node.js如何使用officecrypto-tool實(shí)現(xiàn)讀取加密的Excel和Word文檔的功能,感興趣的小伙伴可以跟隨小編一起了解一下2023-09-09Node.js如何實(shí)現(xiàn)MySQL數(shù)據(jù)庫連接池
文章介紹了Node.js中實(shí)現(xiàn)MySQL數(shù)據(jù)庫連接池的方法,通過預(yù)先建立和管理數(shù)據(jù)庫連接對象,優(yōu)化數(shù)據(jù)庫連接的使用效率2024-11-11使用koa2創(chuàng)建web項(xiàng)目的方法步驟
這篇文章主要介紹了使用koa2創(chuàng)建web項(xiàng)目的方法步驟,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2019-03-03使用Node.js實(shí)現(xiàn)ORM的一種思路詳解(圖文)
這篇文章主要介紹了用Node.js實(shí)現(xiàn)ORM的一種思路詳解(圖文),需要的朋友可以參考下2017-10-10nodejs基于WS模塊實(shí)現(xiàn)WebSocket聊天功能的方法
這篇文章主要介紹了nodejs基于WS模塊實(shí)現(xiàn)WebSocket聊天功能的方法,結(jié)合實(shí)例形式分析了nodejs使用WS模塊進(jìn)行WebSocket通信實(shí)現(xiàn)聊天功能的具體操作技巧,需要的朋友可以參考下2018-01-01node.js?readline和line-reader逐行讀取文件方法
Readline是Node的原生模塊。它是專門為從任何可讀流逐行讀取內(nèi)容而開發(fā)的。它可用于從命令行讀取數(shù)據(jù),line-reader模塊是Node.js中逐行讀取文件的開源模塊。它不是本地模塊,所以你需要使用npm(節(jié)點(diǎn)包管理器)安裝它2022-10-10