体验 Node.js 的 net 模块
admin
2024-03-10 04:54:19
0

1. 创建客户端

模拟 http 请求,将接收到的响应体内容原样输出,接收完毕后,关闭连接

1.1 最终效果

接收到的数据:

解析后的数据:

1.2 流程说明

初始化:

  • 创建客户端 const client = net.createConnection(options, cb),其中回调 cb 会在连接成功时触发一次。
  • 注册监听函数 client.on('data', () => {})
  • 发送请求 client.write()

data 事件处理逻辑:

data 事件在每次接收到来自服务端的消息时,都会触发。

第一次接收到响应消息时,解析出响应行、响应头信息。之后每次监听到响应消息,都是剩下的响应体的消息,只要不断拼接到响应体中即可。

每次接收完消息后,都要判断来自服务端的消息是否已经接收完毕了,如果接收完毕了,则需要断开链接。

断开连接:

判断是否需要断开连接,可以借助解析出的响应头的 Content-Length 字段来判断。

  • 「1」服务端返回的响应体消息的总字节数:Content-Length 的值
  • 「2」当前接收到的消息的总字节数:Buffer.from(目前接收到的服务端消息体字符串, 'utf-8').byteLength

一旦 「2」 >= 「1」,那么意味着服务端吐给我们的数据,我们都拿到了,此时需要断开可服务端的连接。

客户端主动断开连接的方法:client.end()

1.3 源码

const net = require("net");
const responseData = {line: null, // 响应行header: null, // 响应头body: "", // 响应体
};
const separator = "\r\n"; // 分隔符// 创建客户端
const client = net.createConnection({port: 80, // HTTP 协议,默认端口 80host: "www.baidu.com", // default val => 'localhost'},() => {// 连接成功之后的回调console.log("连接成功~");}
);// 发送请求
client.write(`GET / HTTP/1.1
Connection: keep-alive
Host: www.baidu.com`);// 监听响应
client.on("data", (chunk) => {console.log("chunk => ", chunk.toString("utf-8"));if (!responseData.line) { // 第一次收到的响应消息// 解析第一次接收到的 chunk 获取到响应行、响应头以及响应体的部分信息parseResponse(chunk.toString("utf-8"));} else { // 非第一次接收到的响应消息responseData.body += chunk.toString("utf-8");}isOver();
});// 监听断开
client.on("close", () => {console.log("连接断开~");
});/*** 解析响应消息* @param {String} response 响应消息*/
function parseResponse(response) {const lineEndIndex = response.indexOf(separator); // => 响应行的结束位置const headerEndIndex = response.indexOf(separator + separator); // => 响应头的结束位置const lineStr = response.slice(0, lineEndIndex);const headerStr = response.slice(lineEndIndex + 2, headerEndIndex);const bodyStr = response.slice(headerEndIndex + 4);const lineArr = lineStr.split(" ");const headerArr = headerStr.split(separator);// 响应行responseData.line = {HTTPVersion: lineArr[0], // => 协议版本StatusCode: lineArr[1], // => 状态码ReasonPhrase: lineArr[2], // => 状态码描述};// 响应头responseData.header = headerArr.map((it) => {const keyEndIndex = it.indexOf(": "),key = it.slice(0, keyEndIndex),val = it.slice(keyEndIndex + 2);return [key, val];}).reduce((a, b) => {a[b[0]] = b[1];return a;}, {});// 响应体responseData.body = bodyStr;
}/*** 判断来自服务器的消息是否已经接收完毕*/
function isOver() {const contentLength = +responseData.header["Content-Length"],curLen = Buffer.from(responseData.body).byteLength;// 消息接收完毕if (curLen >= contentLength) {client.end(); // 关闭连接}
} 

主要解决几个问题:

  • 如何创建客户端,建立与服务端的链接
  • 如何使用客户端发送 HTTP 请求
  • 如何拿到服务端返回的 HTTP 响应数据
  • 如何判断服务端响应的内容是否都接收完毕,并在接收完毕之后,关闭连接

1.4 补充

响应消息中,有些字段是重复的,暂时还不理解这些重复的 key 是干啥的,使用上述逻辑处理的最终结果是,后者覆盖前者。

2. 创建服务端

模拟 HTTP 服务器,使用浏览器访问该服务,得到一个静态资源

2.1 最终效果

http://localhost:2155/ 使用浏览器访问本地搭建的一个服务,可以获取到我们返回的静态资源。

2.2 流程说明

初始化:

  • 创建服务端:const localServer = net.createServer()
  • 监听 2155 端口:localServer.listen(2155, () => {})* 注意:回调函数仅会在客户端连接 2155 端口成功时触发一次。
  • 监听来自客户端的连接请求:localServer.on("connection", (socket) => {})* 注意:每次有客户端连接,都会触发 connection 事件。每个客户端都对应一个 socket,它们之间是相互独立的。

处理 socket:

  • 注册监听事件:socket.on("data", (chunk) => {}) 注册 data 事件,每次接收到来自客户端的数据时触发
  • 注册 end 事件,连接断开时触发:socket.on('end', () => {})

准备响应的数据:

  • 响应头
const headBuffer = Buffer.from(
`HTTP/1.1 200 OK
Content-Type: image/jpeg`, "utf-8"); 
  • 响应体,读取静态文件资源(buffer 格式)稍后作为 响应体 返回:const bodyBuffer = await fs.promises.readFile(path.resolve(__dirname, './xxx'))

响应数据:

  • 拼接响应头和响应体,并返回给客户端:socket.write(Buffer.concat([headBuffer, bodyBuffer]))

断开连接:

socket.end()

2.3 源码

const net = require("net");
const path = require("path");
const fs = require("fs");
const localServer = net.createServer();localServer.listen(2155, () => {console.log("开始监听 2155 端口");
}); // => 监听 2155 端口localServer.on("connection", (socket) => {console.log("有客户端连接到该服务了");socket.on("data", async (chunk) => {console.log("接收到来自客户端的数据:", chunk.toString("utf-8"));const headBuffer = Buffer.from(`HTTP/1.1 200 OK
Content-Type: image/jpeg`,"utf-8");// 读取本地头像文件 avatar.jpegconst filename = path.resolve(__dirname, "./avatar.jpeg");// const filename = path.resolve(__dirname, "./index.html");const bodyBuffer = await fs.promises.readFile(filename);socket.write(Buffer.concat([headBuffer, bodyBuffer]));socket.end();});socket.on("end", () => {console.log("连接关闭了");});
}); 

最后

整理了一套《前端大厂面试宝典》,包含了HTML、CSS、JavaScript、HTTP、TCP协议、浏览器、VUE、React、数据结构和算法,一共201道面试题,并对每个问题作出了回答和解析。

有需要的小伙伴,可以点击文末卡片领取这份文档,无偿分享

部分文档展示:



文章篇幅有限,后面的内容就不一一展示了

有需要的小伙伴,可以点下方卡片免费领取

相关内容

热门资讯

广州房贷“减负”:多家银行“商... 每经记者|陈荣浩 每经编辑|黄胜 5月26日,广州住房公积金管理中心发布《广州商业性个人住房贷款转...
2026年北京高端家庭服务市场... 引言 步入2026年,北京家庭服务市场正经历从“劳动力密集型”向“专业能力驱动型”的深刻变革。随着高...
亚马逊员工刷AI数据致算力成本... 亚马逊近日关闭了一项内部AI使用量排行榜。此前员工为冲排名、刻意刷高AI调用量,导致公司算力成本激增...
聚势AI向新 深圳加速构建产融... 上证报中国证券网讯(记者 徐潇潇)5月28日,记者在2026年第四届香蜜湖财富管理论坛了解到,本届论...
阿里语音大模型登顶Speech... 【太平洋科技快讯】据报道,在全球权威AI评测平台Artificial Analysis的Speech...
AI产业保险落地,真就利好赛道... 本篇为大家准备了5条要闻,都是近期大家关注度比较高的内容,看完能理清近期的产业和市场动向。一、要闻导...
智象未来CEO:多模态模型To... 【CNMO科技消息】5月28日,据36氪报道,智象未来CEO梅涛称:智象未来做的是全球唯三、能够达到...
原创 人... 国家统计局2026年1月19日交出了一份让人心情复杂的成绩单。2025年全年出生人口792万人,人口...
咽东西总觉得有个坎儿?半个月还... 很多人都有过这样的经历:吃饭时,感觉食物经过胸口某个地方“顿了一下”或“卡了一下”,好像被什么挡住似...
原创 A... A股市场正上演一场冰与火之歌。 一边是通信、电子等科技板块高歌猛进,另一边是消费、金融等传统行业持续...
原创 5... 2026年5月28日,国内黄金价格继续走低,现货黄金跌至每克986元,中国黄金的基础金价报985.7...
抖音生活服务开放日:打造所见即... 5月27日,抖音生活服务举办“2026年服务体验与治理开放日”活动,分享平台在消费者权益保护与体验提...
原创 珀... “双向奔赴”的效果,拭目以待。 作者 | 方璐 编辑丨于婞 来源 | 野马财经 “618”年中大促的...
创投圈“围抢项目”白热化:“上... 记者 老盈盈 只是经过短短5个月,周政宁所在的慧科科创投资于2025年12月底初次接触的一家具身智能...
原创 5... 长鑫科技,这家名字听着有点陌生的公司,2026年一季度每天净赚2.75亿元,全年利润预计超千亿。 上...
首度失守200日牛熊分界线,黄... 这轮黄金的下跌,处处透露着不寻常。 从昨晚开始,市场就已经出现了明显异动。油价在跌,黄金居然也在跌,...
阿维塔招股书失效,上市日期将推... 据港交所官网披露,阿维塔科技(重庆)股份有限公司(以下简称“阿维塔”)的招股书已经失效。按照港交所规...
黄仁勋,入职清华! 来源:市场资讯 (来源:新智元) 新智元报道 【新智元导读】英伟达 CEO 黄仁勋即将加入清华大学...
新质策源导刊丨朱钧宇:低空经济... 朱钧宇 ——访赛迪研究院产业政策研究所研究室主任朱钧宇 图片来源/摄图网授权 ■中国经济时报记者 ...