muduo库chat server对TCP粘包问题的处理


粘包问题的最本质原因在与接收对等方无法分辨消息与消息之间的边界在哪。我们通过使用某种方案给出边界,例如:

  • 发送定长包。如果每个消息的大小都是一样的,那么在接收对等方只要累计接收数据,直到数据等于一个定长的数值就将它作为一个消息。
  • 包尾加上\r\n标记。FTP协议正是这么做的。但问题在于如果数据正文中也含有\r\n,则会误判为消息的边界。
  • 包头加上包体长度。包头是定长的4个字节,说明了包体的长度。接收对等方先接收包体长度,依据包体长度来接收包体。
  • 使用更加复杂的应用层协议
muduo库采用的是第三种方案,包头存放包体长度。

实现比较简单,直接上代码:
//在该函数中解析消息,是ChatServer首先调用的函数,通过定长包的形式解决了TCP的粘包问题
  void onMessage(const muduo::net::TcpConnectionPtr& conn,
                 muduo::net::Buffer* buf,
                 muduo::Timestamp receiveTime)
  {
    while (buf->readableBytes() >= kHeaderLen) // kHeaderLen == 4 //判断是否超过包头,如果包头都超不过,那半个消息都算不上
    {
      // FIXME: use Buffer::peekInt32()
      const void* data = buf->peek();  //偷看一下readable的当前首地址
      int32_t be32 = *static_cast<const int32_t*>(data); // 
### muduo 中的 TCP 服务器实现与用法 muduo 是一个高性能的 C++ 网络,专为 Linux 平台设计,支持多线程事件驱动模型。它封装了许多复杂的底层操作,使得开发者能够专注于业务逻辑而不是网络编程细节。以下是关于 muduoTCP 服务器实现和使用的详细介绍。 #### 泛型 TCP 服务器的设计 muduo 提供了一个通用的 `TcpServer` 类,用于构建基于 TCP 协议的服务端应用程序。该类的核心设计理念是通过事件循环机制处理连接请求、数据读写以及错误管理等功能[^1]。具体来说: - **EventLoop**: 这是一个核心组件,负责监听并分发各种 I/O 和定时器事件。 - **Channel**: 表示文件描述符上的事件注册状态及其回调函数绑定关系。 - **Acceptor**: 负责接受新的客户端连接,并将其交给主线程或其他工作线程进行进一步处理。 #### 创建一个简单的 TCP 服务器实例 下面展示如何利用 muduo 构建一个基本的回显服务器(Echo Server),即接收到的数据会被原样返回给发送方。 ```cpp #include "muduo/net/EventLoop.h" #include "muduo/net/TcpServer.h" using namespace muduo; using namespace muduo::net; void onConnection(const TcpConnectionPtr& conn) { LOG_INFO << (conn->connected() ? "UP" : "DOWN"); } void onMessage(const TcpConnectionPtr& conn, Buffer* buf, Timestamp time) { std::string msg(buf->retrieveAllAsString()); conn->send(msg); } int main() { EventLoop loop; InetAddress listenAddr(2023); TcpServer server(&loop, listenAddr, "EchoServer"); server.setConnectionCallback(onConnection); server.setMessageCallback(onMessage); server.start(); loop.loop(); return 0; } ``` 上述代码片段展示了几个重要部分: - 设置地址 (`InetAddress`) 及初始化 `TcpServer`. - 定义连接建立或断开时触发的动作 (`onConnection`)。 - 处理消息接收后的响应行为 (`onMessage`)【^2】。 #### 关键概念解析 为了更好地理解以上代码的功能,这里解释一些重要的术语和技术点: - **Buffer**: 数据缓冲区对象,在异步通信场景下尤为重要;它可以有效减少频繁内存分配带来的性能损耗。 - **Timestamp**: 时间戳结构体,用来记录特定时刻以便后续统计或者超时判断之用。 - **TcpConnectionPtr**: 对应于每条已建立成功的TCP链路的一个智能指针形式表示。 此外值得注意的是,muduo内部大量采用了RAII原则来自动释放资源从而降低潜在泄漏风险;同时也提供了丰富的日志工具帮助调试复杂问题. #### 性能优化建议 虽然上面的例子已经足够简单易懂,但在实际生产环境中还需要考虑更多因素才能达到最佳效果。例如可以通过调整线程池大小匹配硬件能力、启用零拷贝技术提高吞吐量等方式来进行针对性改进[^4]. ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值