
在网络编程中,tcp(传输控制协议)提供的是一个可靠的、面向连接的字节流服务。这意味着数据在传输过程中被视为一个连续的字节序列,而非离散的“消息”或“数据包”。当服务器使用socket.write()发送数据时,它仅仅是将字节推送到网络缓冲区,并不能自动通知接收方“一个消息已经发送完毕”。
C语言客户端的recv()函数,其行为是阻塞式的,直到有数据可用、连接关闭或发生错误。在提供的GetData函数实现中,recv被放置在一个while循环中,并持续调用,直到recv返回0(表示对端关闭连接)或-1(表示错误)。
socket.write(Buffer.from("123")) 的阻塞原因: 当Node.js服务器使用socket.write()发送数据后,它并没有关闭连接。客户端的recv()函数会接收到这些字节,但由于服务器没有发出连接关闭的信号,recv会认为可能还有更多数据即将到来,因此它会持续阻塞在while ((bytes_read = recv(...)) > 0)循环中,等待更多数据,从而导致连接“卡住”。
socket.end(Buffer.from("123")) 不阻塞的原因:socket.end()不仅发送了数据,更重要的是,它向对端发送了一个FIN(Finish)包,表示发送方已无更多数据要发送,并请求关闭连接的写入端。当客户端的recv()接收到FIN包时,它会返回0,这使得GetData函数中的while循环得以终止,从而避免了阻塞。然而,这种方式的缺点是每次发送数据后都需要重新建立连接以进行后续读取,这在多数应用场景中是不可接受的。
由于TCP是字节流,客户端无法仅凭recv()的返回值来判断一个“逻辑消息”的结束。为了实现非阻塞且持续的通信,我们需要在应用层协议中明确定义消息的边界。以下是几种常用的方法:
固定长度消息: 双方约定每条消息的长度都是固定的。客户端每次读取固定数量的字节即可。
立即学习“C语言免费学习笔记(深入)”;
长度前缀消息: 在发送实际数据之前,先发送一个固定长度的字段来表示后续数据的长度。
特定分隔符消息: 在每条消息的末尾添加一个或多个特殊字符作为分隔符。
在多数场景下,长度前缀消息是最常用且推荐的方法。下面以长度前缀为例,展示如何修改服务器和客户端代码以实现可靠通信。
我们将使用一个4字节的无符号整数作为长度前缀,表示后续消息体的字节数。
服务器在发送数据时,首先计算消息体的长度,将其写入一个4字节的Buffer,然后将这个长度Buffer与消息体Buffer拼接后发送。
const net = require('net');
const server = net.createServer((socket) => {
console.log('Client connected.');
// 监听客户端数据(如果客户端有发送数据)
socket.on('data', (data) => {
console.log(`Received from client: ${data.toString()}`);
});
// 发送一个带有长度前缀的消息
const sendMessage = (messageString) => {
const messageBuffer = Buffer.from(messageString, 'utf8');
const lengthBuffer = Buffer.alloc(4); // 4字节表示长度
// 将消息长度写入长度Buffer,使用大端字节序 (BE - Big Endian)
// C客户端通常使用网络字节序,即大端字节序
lengthBuffer.writeUInt32BE(messageBuffer.length, 0);
// 将长度Buffer和消息Buffer拼接后发送
socket.write(Buffer.concat([lengthBuffer, messageBuffer]));
console.log(`Sent: "${messageString}" (Length: ${messageBuffer.length})`);
};
// 示例:发送多条消息
sendMessage("Hello from Node.js server!");
setTimeout(() => {
sendMessage("This is another message.");
}, 1000);
setTimeout(() => {
sendMessage("And a third one, longer than the others to demonstrate variable length.");
}, 2000);
socket.on('end', () => {
console.log('Client disconnected.');
});
socket.on('error', (err) => {
console.error('Socket error:', err);
});
});
const PORT = 12345;
server.listen(PORT, () => {
console.log(`Node.js server listening on port ${PORT}`);
});客户端需要分两步读取:首先读取4字节的长度前缀,然后根据这个长度再读取相应数量的字节作为消息体。
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h> // For ntohl (network to host long)
#include <sys/socket.h>
// 辅助函数:确保读取到指定数量的字节
// 返回值:实际读取的字节数,0表示连接关闭,-1表示错误
ssize_t read_n_bytes(int fd, void *buf, size_t n) {
size_t total_read = 0;
ssize_t bytes_read;
while (total_read < n) {
bytes_read = recv(fd, (char *)buf + total_read, n - total_read, 0);
if (bytes_read <= 0) { // 0 for disconnect, -1 for error
return bytes_read;
}
total_read += bytes_read;
}
return total_read;
}
// 接收一个完整的消息(带有长度前缀)
char *GetData(int socket_fd) {
uint32_t message_length_net; // 用于存储网络字节序的消息长度
// 1. 读取4字节的长度前缀
ssize_t res = read_n_bytes(socket_fd, &message_length_net, sizeof(message_length_net));
if (res <= 0) {
// 连接关闭或发生错误
if (res == 0) {
printf("Server disconnected.\n");
} else {
perror("Failed to read message length");
}
return NULL;
}
// 将网络字节序转换为本机字节序
uint32_t message_length = ntohl(message_length_net);
printf("Expected message length: %u bytes\n", message_length);
if (message_length == 0) {
// 收到一个空消息
char *empty_buffer = (char *)malloc(1);
if (empty_buffer) *empty_buffer = '\0';
return empty_buffer;
}
// 2. 根据读取到的长度分配缓冲区
char *buffer = (char *)malloc(message_length + 1); // +1 用于字符串的null终止符
if (buffer == NULL) {
perror("Failed to allocate buffer for message");
return NULL;
}
// 3. 读取消息体
res = read_n_bytes(socket_fd, buffer, message_length);
if (res <= 0) {
perror("Failed to read message body");
free(buffer);
return NULL;
}
buffer[message_length] = '\0'; // null终止字符串
return buffer;
}
// 简单的客户端连接函数
int connect_to_server(const char *ip, int port) {
int sock = 0;
struct sockaddr_in serv_addr;
if ((sock = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
perror("Socket creation error");
return -1;
}
serv_addr.sin_family = AF_INET;
serv_addr.sin_port = htons(port);
if (inet_pton(AF_INET, ip, &serv_addr.sin_addr) <= 0) {
perror("Invalid address/ Address not supported");
return -1;
}
if (connect(sock, (struct sockaddr *)&serv_addr, sizeof(serv_addr)) < 0) {
perror("Connection Failed");
return -1;
}
printf("Connected to server.\n");
return sock;
}
int main() {
int client_socket = connect_to_server("127.0.0.1", 12345); // 连接到Node.js服务器
if (client_socket < 0) {
return 1;
}
char *received_data;
// 循环接收消息,直到服务器关闭连接或发生错误
while ((received_data = GetData(client_socket)) != NULL) {
printf("Received message: \"%s\"\n", received_data);
free(received_data); // 释放动态分配的内存
}
close(client_socket); // 关闭套接字
printf("Client disconnected.\n");
return 0;
}通过在应用层定义明确的消息边界,我们可以克服TCP字节流的特性,实现Node.js服务器和C语言客户端之间高效、可靠且非阻塞的双向通信。关键在于理解recv的阻塞行为和TCP的流式本质,并据此设计合适的消息解析逻辑。
以上就是Node.js与C语言网络通信:理解TCP流与消息边界处理的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号