在 Node.js 中使用 HTTP Agent 实现 keep-alive

不带 Keep-Alive 的请求

我们都知道,浏览器中多个 HTTP 请求是可以通过请求头 connection: keep-alive; 来复用 TCP 连接的,但是在 Node.js 中怎么实现呢?

比如,下面是一段简单的发起请求的代码:

js 复制代码
// server.js
const http = require('http')

http
  .createServer((req, res) => {
    res.writeHead(200)
    res.end('Hello World')
  })
  .listen(3000)

// client.js
const http = require('http')

function request() {
  http
    .request(
      {
        host: 'localhost',
        port: 3000,
        pathname: '/',
      },
      (res) => {
        let buffer = []
        res.on('data', (chunk) => {
          buffer.push(chunk)
        })
        res.on('end', () => {
          console.log(buffer.toString())
        })
      }
    )
    .end()
}

request()

setTimeout(() => {
  request()
}, 1000)

当我们用 wireshard 抓包分析师,我们可以两次请求的 client 端口是不同的,并且有两个"三次握手"的过程:

使用 http.Agent 发送带有 Keep-Alive 的请求

现在,让我们使用 http.Agent 来实现 keep-alive 请求,我们只需要加少量的代码:

js 复制代码
const agent = new http.Agent({
  keepAlive: true,
})

function request() {
  http
    .request(
      {
        agent,
        host: 'localhost',
        port: 3000,
        pathname: '/',
      },
      () => {
        // ...
      }
    )
    .end()
}

但是,wireshark 中的结果其实不会发生变化!实际上,我们需要指定 agentmaxSockets 参数:

js 复制代码
const agent = new http.Agent({
  keepAlive: true,
  maxSockets: 1,
})

为什么呢?因为 maxSockets 表示每个 host 所能建立的最大 TCP 连接数。其默认值是 Infinity。如果我们不指定它的值,那每个 HTTP 请求都会建立一个 TCP 连接。

接下来,我们修改一下代码:

js 复制代码
setTimeout(() => {
  request()
}, 10000) // 1000 -> 10000

wireshark 抓包显示:

keep-alive 又不生效了!并且我们可以看到服务端 5s 左右后发送了一个 FIN 的包。难道有什么参数可以控制 keep-alive 的超时时间么?确实,也就是 keepAliveTimeout,我们把它设置为 10s:

js 复制代码
const http = require('http')

const server = http
  .createServer((req, res) => {
    res.writeHead(200)
    res.end('Hello World')
  })
  .listen(3000)

server.keepAliveTimeout = 10000

现在 keep-alive 又可以工作了。

求关注公众号"前端游"

相关推荐
Bigger1 小时前
Bun 能上生产吗?我的实战结论
前端·node.js·bun
不会敲代码14 小时前
从零搭建 RAG 电子书智能问答系统:天龙八部 × Milvus × LangChain
langchain·node.js·llm
donecoding1 天前
nrm、corepack、npm registry 三者的爱恨情仇
前端·node.js·前端工程化
明月_清风1 天前
关于node 模块化的现状与未来
前端·node.js
李白的天不白2 天前
webpack 与 webpack-cli 版本匹配问题
前端·webpack·node.js
李白的天不白2 天前
webpack 与axios 版本冲突问题
前端·webpack·node.js
donecoding3 天前
pnpm 全局包与 nvm 的真相:命令永在,运行时随缘
node.js·claude
老蒋每日coding3 天前
Node.js 安装指南(Mac 版本)
macos·node.js
吴声子夜歌3 天前
Node.js——JSON-Server轻量级RESTful API
node.js·json·restful·json-server