日韩无码专区无码一级三级片|91人人爱网站中日韩无码电影|厨房大战丰满熟妇|AV高清无码在线免费观看|另类AV日韩少妇熟女|中文日本大黄一级黄色片|色情在线视频免费|亚洲成人特黄a片|黄片wwwav色图欧美|欧亚乱色一区二区三区

RELATEED CONSULTING
相關(guān)咨詢(xún)
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時(shí)間:8:30-17:00
你可能遇到了下面的問(wèn)題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營(yíng)銷(xiāo)解決方案
Node.jsHTTP模塊的內(nèi)存泄露問(wèn)題

很久沒(méi)有逛社區(qū)了,晚上回來(lái)看了一下最近的情況,突然看到一個(gè)內(nèi)存泄露問(wèn)題,作為一個(gè) APM 開(kāi)發(fā)者,自然想分析其中的原因。

公司主營(yíng)業(yè)務(wù):成都做網(wǎng)站、網(wǎng)站建設(shè)、外貿(mào)營(yíng)銷(xiāo)網(wǎng)站建設(shè)、移動(dòng)網(wǎng)站開(kāi)發(fā)等業(yè)務(wù)。幫助企業(yè)客戶(hù)真正實(shí)現(xiàn)互聯(lián)網(wǎng)宣傳,提高企業(yè)的競(jìng)爭(zhēng)能力。創(chuàng)新互聯(lián)是一支青春激揚(yáng)、勤奮敬業(yè)、活力青春激揚(yáng)、勤奮敬業(yè)、活力澎湃、和諧高效的團(tuán)隊(duì)。公司秉承以“開(kāi)放、自由、嚴(yán)謹(jǐn)、自律”為核心的企業(yè)文化,感謝他們對(duì)我們的高要求,感謝他們從不同領(lǐng)域給我們帶來(lái)的挑戰(zhàn),讓我們激情的團(tuán)隊(duì)有機(jī)會(huì)用頭腦與智慧不斷的給客戶(hù)帶來(lái)驚喜。創(chuàng)新互聯(lián)推出聞喜免費(fèi)做網(wǎng)站回饋大家。

問(wèn)題

下面介紹一下具體的問(wèn)題??匆幌?demo。

const http = require('http')

async function main () {
  let i = 0
  while (true) {
    if (i % 100 === 0) {
      global.gc()
    }

    if (i % 10000 === 0) {
      console.log(process.memoryUsage().heapUsed)
    }

    http.createServer((req, res) => {})
    i++
  }
}

main()

Node.js v20.3.1 下執(zhí)行上面代碼(node --expose-gc demo.js)輸出如下。

2681120
11409488
19632792
28038016
36438104

可以看到內(nèi)存不斷在增長(zhǎng)。下面來(lái)分析這個(gè)問(wèn)題。

分析

const http = require('http');
const v8 = require('v8');

for (i = 0; i < 1000; i++) {
    http.createServer((req, res) => {});
}
v8.writeHeapSnapshot('memory-leaky.heapsnapshot');

采集的快照如下。

圖片

可以看到,Server 對(duì)象沒(méi)有被釋放??匆幌率钦l(shuí)引用了它。

圖片

是定時(shí)器引用了 Server 對(duì)象,我們看一下定時(shí)器對(duì)象又是被誰(shuí)引用了。

圖片

有一個(gè)關(guān)鍵的變量 connectionsCheckingInterval,到 Node.js 源碼里看一下,最終發(fā)現(xiàn)是 Server 初始化時(shí)創(chuàng)建的。

function Server(options, requestListener) {
  setupConnectionsTracking(this);
}

function setupConnectionsTracking(server) {
  server[kConnectionsCheckingInterval] = setInterval(checkConnections.bind(server), server.connectionsCheckingInterval).unref();
}

可以看到 checkConnections.bind 返回的匿名函數(shù)持有了 Server,而匿名函數(shù)又被 setInterval 持有了,所以導(dǎo)致 Server 對(duì)象無(wú)法釋放。

修復(fù)

那么如何修復(fù)這個(gè)問(wèn)題呢?修復(fù)這個(gè)問(wèn)題,首先需要了解 setupConnectionsTracking 是做什么的,邏輯如下。

function checkConnections() {
  if (this.headersTimeout === 0 && this.requestTimeout === 0) {
    return;
  }

  const expired = this[kConnections].expired(this.headersTimeout, this.requestTimeout);

  for (let i = 0; i < expired.length; i++) {
    const socket = expired[i].socket;

    if (socket) {
      onRequestTimeout(socket);
    }
  }
}

可以看到,setupConnectionsTracking 是追蹤連接超時(shí),回到我們的測(cè)試?yán)又锌梢园l(fā)現(xiàn),我們并沒(méi)有執(zhí)行 listen,也就是說(shuō),Server 對(duì)象并不會(huì)處理連接,那么也就沒(méi)有連接需要追蹤,所以修復(fù)方式就是把調(diào)用 setupConnectionsTracking 的時(shí)機(jī)延遲到 listen 成功時(shí),修復(fù)代碼大致如下。

function Server(options, requestListener) {
  this.on('listening', () => {
    setupConnectionsTracking(this);
  });
}

修改源碼重新編譯后測(cè)試結(jié)果如下。

3653552
4002680
3753400
3762976
3773088

可以看到內(nèi)存已經(jīng)不會(huì)增長(zhǎng)了,采集快照也可以看到不會(huì)再存在大量 Server 對(duì)象。

總結(jié)

這個(gè)例子雖然看起來(lái)有點(diǎn)不常見(jiàn),用法也很怪異,但是從側(cè)面說(shuō)明了雖然 JS 自帶 GC,但是因?yàn)檫壿?/ 引用關(guān)系復(fù)雜,還是很容易出現(xiàn)內(nèi)存泄露問(wèn)題,所以寫(xiě)代碼時(shí)還是需要注意,具體的 issue 可以參考 https://github.com/nodejs/node/issues/48604。


當(dāng)前標(biāo)題:Node.jsHTTP模塊的內(nèi)存泄露問(wèn)題
文章分享:http://m.5511xx.com/article/cospcoo.html