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道面试题,并对每个问题作出了回答和解析。
有需要的小伙伴,可以点击文末卡片领取这份文档,无偿分享
部分文档展示:
文章篇幅有限,后面的内容就不一一展示了
有需要的小伙伴,可以点下方卡片免费领取