成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Node.js HTTP 模塊的內存泄露問題

開發 前端
setupConnectionsTracking 是追蹤連接超時,回到我們的測試例子中可以發現,我們并沒有執行 listen,也就是說,Server 對象并不會處理連接,那么也就沒有連接需要追蹤,所以修復方式就是把調用 setupConnectionsTracking 的時機延遲到 listen 成功時,修復代碼大致如下。

很久沒有逛社區了,晚上回來看了一下最近的情況,突然看到一個內存泄露問題,作為一個 APM 開發者,自然想分析其中的原因。

問題

下面介紹一下具體的問題。看一下 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 下執行上面代碼(node --expose-gc demo.js)輸出如下。

2681120
11409488
19632792
28038016
36438104

可以看到內存不斷在增長。下面來分析這個問題。

分析

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

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

采集的快照如下。

圖片圖片

可以看到,Server 對象沒有被釋放。看一下是誰引用了它。

圖片圖片

是定時器引用了 Server 對象,我們看一下定時器對象又是被誰引用了。

圖片圖片

有一個關鍵的變量 connectionsCheckingInterval,到 Node.js 源碼里看一下,最終發現是 Server 初始化時創建的。

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

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

可以看到 checkConnections.bind 返回的匿名函數持有了 Server,而匿名函數又被 setInterval 持有了,所以導致 Server 對象無法釋放。

修復

那么如何修復這個問題呢?修復這個問題,首先需要了解 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 是追蹤連接超時,回到我們的測試例子中可以發現,我們并沒有執行 listen,也就是說,Server 對象并不會處理連接,那么也就沒有連接需要追蹤,所以修復方式就是把調用 setupConnectionsTracking 的時機延遲到 listen 成功時,修復代碼大致如下。

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

修改源碼重新編譯后測試結果如下。

3653552
4002680
3753400
3762976
3773088

可以看到內存已經不會增長了,采集快照也可以看到不會再存在大量 Server 對象。

總結

這個例子雖然看起來有點不常見,用法也很怪異,但是從側面說明了雖然 JS 自帶 GC,但是因為邏輯 / 引用關系復雜,還是很容易出現內存泄露問題,所以寫代碼時還是需要注意,具體的 issue 可以參考 https://github.com/nodejs/node/issues/48604。

責任編輯:武曉燕 來源: 編程雜技
相關推薦

2025-01-08 08:47:44

Node.js內存泄露定時器

2021-09-26 05:06:04

Node.js模塊機制

2020-04-15 15:48:03

Node.jsstream前端

2025-05-26 00:31:31

2019-12-17 11:40:44

Node.js模塊前端

2017-03-19 16:40:28

漏洞Node.js內存泄漏

2017-03-20 13:43:51

Node.js內存泄漏

2022-01-02 06:55:08

Node.js ObjectWrapAddon

2020-01-03 16:04:10

Node.js內存泄漏

2021-10-03 15:02:50

HTTPNodejs

2014-09-12 10:35:09

Node.jsHTTP 206

2011-12-09 11:16:48

Node.js

2022-06-23 06:34:56

Node.js子線程

2013-11-01 09:34:56

Node.js技術

2015-03-10 10:59:18

Node.js開發指南基礎介紹

2011-09-08 14:07:28

Node.js

2023-06-20 19:35:00

Node.js工具

2017-04-24 08:31:26

Node.jsExpress.jsHTTP

2022-03-02 09:01:30

Node.js v1HTTPNode.j

2021-12-25 22:29:57

Node.js 微任務處理事件循環
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产男女猛烈无遮掩视频免费网站 | 久草在线 | 免费一区二区三区 | 中文字幕亚洲区一区二 | 日韩中文一区 | 久久久久国产一区二区三区不卡 | 黄色免费在线网址 | 成人免费在线网 | 久草新在线| 毛片在线看片 | 韩国成人在线视频 | 欧美极品视频 | 福利片在线看 | 三区四区在线观看 | 夜夜草视频 | 日本激情一区二区 | 亚洲一二三在线观看 | 99久久免费精品国产男女高不卡 | 狠狠干天天干 | 最新国产精品视频 | 国产精品视频一二三 | 在线视频一区二区 | 中文字幕一区二区三区不卡 | 国产九九九九 | 黄色av观看| 男女视频91 | 香蕉一区 | 一区二区三区欧美在线 | 9999在线视频 | 粉嫩在线 | 亚洲高清视频一区二区 | 黄色大片在线视频 | com.色.www在线观看 | 亚洲一区二区三区 | 干干干操操操 | 亚洲午夜精品一区二区三区他趣 | 美女在线一区二区 | 国产日韩欧美精品一区二区 | 香蕉视频一区二区 | 欧美一区二区三区,视频 | 91精品国产综合久久福利软件 |