0

0

Node.js与C语言网络通信:理解TCP流与消息边界处理

霞舞

霞舞

发布时间:2025-07-21 14:46:01

|

471人浏览过

|

来源于php中文网

原创

Node.js与C语言网络通信:理解TCP流与消息边界处理

本文旨在解决Node.js服务器端使用socket.write()与C语言客户端使用recv()进行通信时遇到的连接阻塞问题。核心在于理解TCP协议作为字节流的特性,而非消息导向。文章将解释为何socket.write()会导致recv()阻塞,而socket.end()则不会,并提供通过定义消息边界(如长度前缀)来构建可靠、非阻塞通信机制的专业教程与示例代码。

TCP协议的字节流特性

网络编程中,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()的返回值来判断一个“逻辑消息”的结束。为了实现非阻塞且持续的通信,我们需要在应用层协议中明确定义消息的边界。以下是几种常用的方法:

  1. 固定长度消息: 双方约定每条消息的长度都是固定的。客户端每次读取固定数量的字节即可。

    立即学习C语言免费学习笔记(深入)”;

    • 优点: 实现简单。
    • 缺点: 灵活性差,消息长度必须固定或填充,可能浪费带宽。
  2. 长度前缀消息: 在发送实际数据之前,先发送一个固定长度的字段来表示后续数据的长度。

    • 优点: 灵活,效率较高。
    • 缺点: 需要处理字节序(大小端)问题。
  3. 特定分隔符消息: 在每条消息的末尾添加一个或多个特殊字符作为分隔符。

    Cogram
    Cogram

    使用AI帮你做会议笔记,跟踪行动项目

    下载
    • 优点: 易于理解和实现。
    • 缺点: 如果消息内容本身包含分隔符,需要进行转义处理。

在多数场景下,长度前缀消息是最常用且推荐的方法。下面以长度前缀为例,展示如何修改服务器和客户端代码以实现可靠通信。

示例实现:长度前缀协议

我们将使用一个4字节的无符号整数作为长度前缀,表示后续消息体的字节数。

Node.js 服务器端实现

服务器在发送数据时,首先计算消息体的长度,将其写入一个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}`);
});

C 语言客户端实现

客户端需要分两步读取:首先读取4字节的长度前缀,然后根据这个长度再读取相应数量的字节作为消息体。

#include 
#include 
#include 
#include 
#include  // For ntohl (network to host long)
#include 

// 辅助函数:确保读取到指定数量的字节
// 返回值:实际读取的字节数,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;
}

注意事项与总结

  1. 字节序(Endianness): 在跨平台通信中,特别是C语言和Node.js(通常运行在小端系统上,但网络协议常用大端),务必注意字节序问题。Node.js的Buffer.writeUInt32BE和C语言的ntohl(network to host long)函数是处理大端网络字节序的关键。
  2. 错误处理: 生产环境的代码需要更健壮的错误处理,包括网络中断、recv返回-1、内存分配失败等情况。
  3. 缓冲区管理: 在C语言中,动态内存分配(malloc/realloc)和释放(free)至关重要,避免内存泄漏。
  4. 分包与粘包: TCP协议可能会将多个write操作的数据合并成一个recv操作接收(粘包),也可能将一个write操作的数据拆分成多个recv操作接收(分包)。我们提供的read_n_bytes辅助函数就是为了处理分包问题,确保每次都能读取到完整的长度前缀或消息体。
  5. 协议设计: 对于更复杂的应用,可能需要设计更完善的应用层协议,例如包含消息类型、校验和等字段,以提高通信的可靠性和灵活性。

通过在应用层定义明确的消息边界,我们可以克服TCP字节流的特性,实现Node.js服务器和C语言客户端之间高效、可靠且非阻塞的双向通信。关键在于理解recv的阻塞行为和TCP的流式本质,并据此设计合适的消息解析逻辑。

相关专题

更多
C语言变量命名
C语言变量命名

c语言变量名规则是:1、变量名以英文字母开头;2、变量名中的字母是区分大小写的;3、变量名不能是关键字;4、变量名中不能包含空格、标点符号和类型说明符。php中文网还提供c语言变量的相关下载、相关课程等内容,供大家免费下载使用。

379

2023.06.20

c语言入门自学零基础
c语言入门自学零基础

C语言是当代人学习及生活中的必备基础知识,应用十分广泛,本专题为大家c语言入门自学零基础的相关文章,以及相关课程,感兴趣的朋友千万不要错过了。

607

2023.07.25

c语言运算符的优先级顺序
c语言运算符的优先级顺序

c语言运算符的优先级顺序是括号运算符 > 一元运算符 > 算术运算符 > 移位运算符 > 关系运算符 > 位运算符 > 逻辑运算符 > 赋值运算符 > 逗号运算符。本专题为大家提供c语言运算符相关的各种文章、以及下载和课程。

348

2023.08.02

c语言数据结构
c语言数据结构

数据结构是指将数据按照一定的方式组织和存储的方法。它是计算机科学中的重要概念,用来描述和解决实际问题中的数据组织和处理问题。数据结构可以分为线性结构和非线性结构。线性结构包括数组、链表、堆栈和队列等,而非线性结构包括树和图等。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

255

2023.08.09

c语言random函数用法
c语言random函数用法

c语言random函数用法:1、random.random,随机生成(0,1)之间的浮点数;2、random.randint,随机生成在范围之内的整数,两个参数分别表示上限和下限;3、random.randrange,在指定范围内,按指定基数递增的集合中获得一个随机数;4、random.choice,从序列中随机抽选一个数;5、random.shuffle,随机排序。

583

2023.09.05

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

519

2023.09.20

c语言get函数的用法
c语言get函数的用法

get函数是一个用于从输入流中获取字符的函数。可以从键盘、文件或其他输入设备中读取字符,并将其存储在指定的变量中。本文介绍了get函数的用法以及一些相关的注意事项。希望这篇文章能够帮助你更好地理解和使用get函数 。

630

2023.09.20

c数组初始化的方法
c数组初始化的方法

c语言数组初始化的方法有直接赋值法、不完全初始化法、省略数组长度法和二维数组初始化法。详细介绍:1、直接赋值法,这种方法可以直接将数组的值进行初始化;2、不完全初始化法,。这种方法可以在一定程度上节省内存空间;3、省略数组长度法,这种方法可以让编译器自动计算数组的长度;4、二维数组初始化法等等。

595

2023.09.22

桌面文件位置介绍
桌面文件位置介绍

本专题整合了桌面文件相关教程,阅读专题下面的文章了解更多内容。

0

2025.12.30

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Rust 教程
Rust 教程

共28课时 | 3.9万人学习

Kotlin 教程
Kotlin 教程

共23课时 | 2.1万人学习

Go 教程
Go 教程

共32课时 | 3.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号