模拟 http 请求,将接收到的响应体内容原样输出,接收完毕后,关闭连接
接收到的数据:
解析后的数据:
初始化:
const client = net.createConnection(options, cb),其中回调 cb 会在连接成功时触发一次。client.on('data', () => {})client.write()data 事件处理逻辑:
data 事件在每次接收到来自服务端的消息时,都会触发。
第一次接收到响应消息时,解析出响应行、响应头信息。之后每次监听到响应消息,都是剩下的响应体的消息,只要不断拼接到响应体中即可。
每次接收完消息后,都要判断来自服务端的消息是否已经接收完毕了,如果接收完毕了,则需要断开链接。
断开连接:
判断是否需要断开连接,可以借助解析出的响应头的 Content-Length 字段来判断。
Content-Length 的值Buffer.from(目前接收到的服务端消息体字符串, 'utf-8').byteLength一旦 「2」 >= 「1」,那么意味着服务端吐给我们的数据,我们都拿到了,此时需要断开可服务端的连接。
客户端主动断开连接的方法:client.end()
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(); // 关闭连接}
}
主要解决几个问题:
响应消息中,有些字段是重复的,暂时还不理解这些重复的 key 是干啥的,使用上述逻辑处理的最终结果是,后者覆盖前者。
模拟 HTTP 服务器,使用浏览器访问该服务,得到一个静态资源
http://localhost:2155/ 使用浏览器访问本地搭建的一个服务,可以获取到我们返回的静态资源。
初始化:
const localServer = net.createServer()localServer.listen(2155, () => {})* 注意:回调函数仅会在客户端连接 2155 端口成功时触发一次。localServer.on("connection", (socket) => {})* 注意:每次有客户端连接,都会触发 connection 事件。每个客户端都对应一个 socket,它们之间是相互独立的。处理 socket:
socket.on("data", (chunk) => {}) 注册 data 事件,每次接收到来自客户端的数据时触发socket.on('end', () => {})准备响应的数据:
const headBuffer = Buffer.from(
`HTTP/1.1 200 OK
Content-Type: image/jpeg`, "utf-8");
const bodyBuffer = await fs.promises.readFile(path.resolve(__dirname, './xxx'))响应数据:
socket.write(Buffer.concat([headBuffer, bodyBuffer]))断开连接:
socket.end()
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道面试题,并对每个问题作出了回答和解析。

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




文章篇幅有限,后面的内容就不一一展示了
有需要的小伙伴,可以点下方卡片免费领取