muduo库TcpConnection模块详解——C++

news2025/5/19 15:38:59

muduo库中的TcpConnection模块详解

TcpConnection是muduo库中处理TCP连接的核心模块,负责管理单个TCP连接的生命周期、数据读写、状态转换以及事件回调。每个TCP连接对应一个TcpConnection对象,其设计体现了高性能、线程安全和灵活回调的特点。


一、核心职责

  1. 连接生命周期管理
    管理TCP连接的建立、活跃状态、关闭过程及销毁。
  2. 数据读写与缓冲
    通过inputBufferoutputBuffer处理数据的接收与发送,应对粘包/半包问题。
  3. 事件回调机制
    提供用户可配置的回调函数(如消息到达、连接关闭、发送完成等)。
  4. 线程安全保证
    确保所有操作在绑定的EventLoop线程中执行,避免竞态条件。

二、关键数据结构与成员

1. 类定义概览
class TcpConnection : public std::enable_shared_from_this<TcpConnection> {
public:
    // 连接状态枚举
    enum State { kConnecting, kConnected, kDisconnecting, kDisconnected };

    // 回调函数类型定义
    using MessageCallback = std::function<void(const TcpConnectionPtr&, Buffer*, Timestamp)>;
    
    using CloseCallback = std::function<void(const TcpConnectionPtr&)>;

private:
    EventLoop* loop_;            // 所属EventLoop(即IO线程)
    State state_;                // 当前连接状态
    std::unique_ptr<Socket> socket_; // 管理的Socket对象
    std::unique_ptr<Channel> channel_; // 关联的Channel,用于事件监听
    Buffer inputBuffer_;         // 接收数据缓冲区
    Buffer outputBuffer_;        // 发送数据缓冲区
    MessageCallback messageCallback_; // 消息到达回调
    CloseCallback closeCallback_;    // 连接关闭回调
};
2. 成员说明
  • loop_
    指向绑定的EventLoop,所有操作必须在该线程执行。
  • state_
    管理连接状态,确保状态转换的合法性(如不允许在已关闭的连接上发送数据)。
  • socket_channel_
    封装底层socket文件描述符及其事件监听,channel_注册到loop_中处理读写事件。
  • inputBuffer_outputBuffer_
    应用层缓冲区,用于暂存接收和待发送的数据。
  • 回调函数
    用户通过setMessageCallback()等接口设置自定义逻辑。

三、生命周期管理

1. 连接建立
  • 创建时机
    TcpServerTcpClient在连接建立时创建TcpConnection对象。
  • 初始化流程
    • 设置socket为非阻塞模式。
    • 绑定ChannelEventLoop,注册可读事件。
    • 状态设为kConnecting,随后通过connectEstablished()转为kConnected
2. 连接关闭
  • 主动关闭
    用户调用shutdown(),发送剩余数据后关闭写端(SHUT_WR)。
  • 被动关闭
    收到FIN包(读返回0)时,触发handleClose()
  • 状态转换
    kConnectedkDisconnectingkDisconnected,最终调用closeCallback_通知上层。
3. 对象销毁
  • 使用shared_ptr管理生命周期,通过shared_from_this()确保回调中对象存活。
  • 连接关闭后,TcpServerTcpClient从连接列表中移除该对象的引用。

四、数据读写流程

1. 数据接收
  • 可读事件处理

    void TcpConnection::handleRead(Timestamp receiveTime) {
        ssize_t n = inputBuffer_.readFd(channel_->fd(), &savedErrno);
        if (n > 0) {
            messageCallback_(shared_from_this(), &inputBuffer_, receiveTime);
        } else if (n == 0) { // 对端关闭连接
            handleClose();
        } else { // 错误处理
            handleError();
        }
    }
    
  • 用户处理逻辑
    通过messageCallback_将数据传递给用户,用户从inputBuffer_解析完整消息。

2. 数据发送
  • 发送入口

    void TcpConnection::send(const std::string& message) {
        if (state_ == kConnected) {
            if (loop_->isInLoopThread()) {
                sendInLoop(message.data(), message.size());
            } else {
                loop_->runInLoop([this, msg = message] { 
                    sendInLoop(msg.data(), msg.size()); 
                });
            }
        }
    }
    
  • 实际发送逻辑

    void TcpConnection::sendInLoop(const void* data, size_t len) {
        if (channel_->isWriting() && outputBuffer_.readableBytes() == 0) {
            // 直接尝试写入socket
            ssize_t n = ::write(channel_->fd(), data, len);
            if (n < 0) { /* 错误处理 */ }
            if (static_cast<size_t>(n) < len) {
                // 剩余数据加入outputBuffer_并注册可写事件
                outputBuffer_.append(static_cast<const char*>(data)+n, len-n);
                channel_->enableWriting();
            }
        } else {
            // 数据直接追加到outputBuffer_
            outputBuffer_.append(data, len);
            if (!channel_->isWriting()) {
                channel_->enableWriting();
            }
        }
    }
    
  • 可写事件处理

    void TcpConnection::handleWrite() {
        ssize_t n = ::write(channel_->fd(), outputBuffer_.peek(), outputBuffer_.readableBytes());
        if (n > 0) {
            outputBuffer_.retrieve(n);
            if (outputBuffer_.readableBytes() == 0) {
                channel_->disableWriting(); // 避免忙等待
                if (state_ == kDisconnecting) {
                    shutdownInLoop(); // 关闭写端
                }
            }
        }
    }
    

五、状态管理与异常处理

1. 状态转换图
kConnecting → kConnected → kDisconnecting → kDisconnected
       ↑_________________________|
  • 关键转换点
    • connectEstablished()kConnectingkConnected
    • shutdown()kConnectedkDisconnecting
    • handleClose()kDisconnectingkDisconnected
2. 异常处理
  • socket错误
    handleError()调用closeCallback_并关闭连接。
  • 对方意外断开
    readFd()返回0时触发handleClose()
  • 发送失败处理
    根据errno判断是否为可恢复错误(如EAGAIN),否则关闭连接。

六、线程安全机制

  • 跨线程调用保护
    所有非IO线程的操作通过loop_->runInLoop()转移到IO线程执行。
  • 智能指针管理
    使用shared_from_this()确保回调执行期间对象不被销毁。
  • 状态变更原子性
    状态state_的修改仅在IO线程中进行,无需加锁。

七、高级功能

  1. TCP_NODELAY选项
    通过setTcpNoDelay()禁用Nagle算法,减少小数据包延迟。
  2. Keep-Alive机制
    setKeepAlive()启用TCP保活探测,检测死连接。
  3. 水位线控制
    设置高水位回调(highWaterMarkCallback_),防止发送缓冲区堆积。

八、典型使用示例

// 创建TcpServer并设置回调
TcpServer server(&loop, InetAddress(9876), "EchoServer");
server.setConnectionCallback([](const TcpConnectionPtr& conn) {
    if (conn->connected()) {
        conn->setTcpNoDelay(true); // 启用TCP_NODELAY
    }
});
server.setMessageCallback([](const TcpConnectionPtr& conn, Buffer* buf, Timestamp) {
    conn->send(buf->retrieveAllAsString()); // 回显数据
});
server.start();
loop.loop();

九、设计思想总结

  1. 资源封装
    将socket、缓冲区、事件监听封装为对象,简化资源管理。
  2. 事件驱动
    基于Reactor模式,通过事件回调实现异步非阻塞IO。
  3. 线程隔离
    每个连接绑定到固定IO线程,避免锁竞争。
  4. 灵活扩展
    用户可通过回调自定义协议处理逻辑(如HTTP、Redis协议)。

通过TcpConnection模块,muduo库实现了高效、稳定的TCP连接管理,支撑了高性能服务器的开发需求。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2378369.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

全局异常处理:如何优雅地统一管理业务异常

在软件开发中&#xff0c;异常处理是保证系统健壮性的重要环节。一个良好的异常处理机制不仅能提高代码的可维护性&#xff0c;还能为使用者提供清晰的错误反馈。本文将介绍如何通过全局异常处理和业务异常统一处理来编写更加优雅的代码。 一、传统异常处理的痛点 1.1 典型问…

动态规划-LCR 166.珠宝的最大价值-力扣(LeetCode)

一、题目解析 frame二维矩阵中每个值代表珠宝的价值&#xff0c;现在从左上角开始拿珠宝&#xff0c;只能向右或向下拿珠宝&#xff0c;到达右下角时停止拿珠宝&#xff0c;要求拿的珠宝价值最大。 二、算法解析 1.状态表示 我们想要知道的是到达[i,j]为位置时的最大价值&am…

JDBC实现模糊、动态与分页查询的详解

文章目录 一. 模糊查询1. Mysql的写法2. JDBC的实现 二. 动态条件查询1. 创建生成动态条件查询sql的方法2. 完整的动态条件查询类以及测试类 三. 分页查询1. 什么是分页查询&#xff1f;2. 分页查询的分类3. MySQL的实现4. JDBC实现4.1. 创建page页4.2. 分页的实现 本章来讲一下…

域环境信息收集技术详解:从基础命令到实战应用

引言 在企业网络环境中&#xff0c;Active Directory (AD)域服务是微软提供的集中式目录服务&#xff0c;用于管理网络中的用户、计算机和其他资源。对于信息安全专业人员来说&#xff0c;熟练掌握域环境信息收集技术至关重要&#xff0c;无论是进行渗透测试、安全评估还是日常…

【C++ Qt】布局管理器

每日激励&#xff1a;“不设限和自我肯定的心态&#xff1a;I can do all things。 — Stephen Curry” &#x1f914;绪论​&#xff1a; 在Qt开发中&#xff0c;界面布局的合理设计是提升用户体验的关键。早期&#xff0c;开发者常采用绝对定位的方式摆放控件&#xff0c;即通…

vscode用python开发maya联动调试设置

如何在VScode里编写Maya Python脚本_哔哩哔哩_bilibili1 包括1&#xff0c;maya的python全面在vscode支持&#xff0c;2&#xff0c;通过mayacode发送到maya&#xff0c;3同步调试 import maya.cmds as cmds 1、让 maya.cmds编译通过 下载Autodesk_Maya_2018_6_Update_DEVK…

SLAM定位常用地图对比示例

序号 地图类型 概述 1 格栅地图 将现实环境栅格化,每一个栅格用 0 和 1 分别表示空闲和占据状态,初始化为未知状态 0.5 2 特征地图 以点、线、面等几何特征来描绘周围环境,将采集的信息进行筛选和提取得到关键几何特征 3 拓扑地图 将重要部分抽象为地图,使用简单的图形表示…

python的漫画网站管理系统

目录 技术栈介绍具体实现截图![在这里插入图片描述](https://i-blog.csdnimg.cn/direct/0ed2084038144499a162b3fb731a5f37.png)![在这里插入图片描述](https://i-blog.csdnimg.cn/direct/a76a091066f74a80bf7ac1be489ae8a8.png)系统设计研究方法&#xff1a;设计步骤设计流程核…

源码安装gperftools工具

源码安装gperftools工具 下载gperftools源码 https://github.com/gperftools/gperftools/releases/download/gperftools-2.16/gperftools-2.16.tar.gz 注&#xff1a;需要下载github上release版本&#xff0c;如果直接下载master分支上源码&#xff0c;将可能出现各种编译报错…

前端脚手架开发指南:提高开发效率的核心操作

前端脚手架通过自动化的方式可以提高开发效率并减少重复工作&#xff0c;而最强大的脚手架并不是现成的那些工具而是属于你自己团队量身定制的脚手架&#xff01;本篇文章将带你了解脚手架开发的基本技巧&#xff0c;帮助你掌握如何构建适合自己需求的工具&#xff0c;并带着你…

搜索引擎工作原理|倒排索引|query改写|CTR点击率预估|爬虫

写在前面 使用搜索引擎是我们经常做的事情&#xff0c;搜索引擎的实现原理。 什么是搜索引擎 搜索引擎是一种在线搜索工具&#xff0c;当用户在搜索框输入关键词时&#xff0c;搜索引擎就会将与该关键词相关的内容展示给用户。比较大型的搜索引擎有谷歌&#xff0c;百度&…

Python实例题:Python自动工资条

目录 Python实例题 题目 python-automatic-payroll-slipPython 自动生成工资条脚本 代码解释 加载文件&#xff1a; 获取表头&#xff1a; 写入表头&#xff1a; 生成工资条&#xff1a; 保存文件&#xff1a; 运行思路 注意事项 Python实例题 题目 Python自动工资…

Function Calling万字实战指南:打造高智能数据分析Agent平台

个人主页&#xff1a;Guiat 归属专栏&#xff1a;科学技术变革创新 文章目录 1. Function Calling&#xff1a;智能交互的新范式1.1 Function Calling 技术概述1.2 核心优势分析 2. 数据分析Agent平台架构设计2.1 系统架构概览2.2 核心组件解析2.2.1 函数注册中心2.2.2 Agent控…

线对板连接器的兼容性问题:为何老旧设计难以满足现代需求?

线对板连接器作为电子设备的核心纽带&#xff0c;正面临前所未有的兼容性挑战。某智能工厂升级生产线时发现&#xff0c;沿用十年的2.54毫米间距连接器&#xff0c;在接入新型工业相机时出现30%的信号丢包率&#xff0c;而切换至0.4毫米超密间距连接器后&#xff0c;数据传输速…

AI517 AI本地部署 docker微调(失败)

本地部署AI 计划使用OLLAMA进行本地部署 修改DNS 访问github 刷新缓存 配置环境变量 OLLAMA安装成功 部署成功 计划使用docker进行微调 下载安装docker 虚拟化已开启 开启上面这些 准备下载ubuntu docker ragflow dify 用git去泡

VR和眼动控制集群机器人的方法

西安建筑科技大学信息与控制工程学院雷小康老师团队联合西北工业大学航海学院彭星光老师团队&#xff0c;基于虚拟现实&#xff08;VR&#xff09;和眼动追踪技术实现了人-集群机器人高效、灵活的交互控制。相关研究论文“基于虚拟现实和眼动的人-集群机器人交互方法” 发表于信…

TiDB 中新 Hash Join 的设计与性能优化

原文来源&#xff1a; https://tidb.net/blog/11667c37 本文作者&#xff1a;徐飞 导读 在数据库管理系统&#xff08;DBMS&#xff09;中&#xff0c;连接操作&#xff08;Join&#xff09;是查询处理的核心环节之一&#xff0c;其性能直接影响到整个系统的响应速度和效率…

1.共享内存(python共享内存实际案例,传输opencv frame)

主进程程序 send.py import cv2 import numpy as np from multiprocessing import shared_memory, resource_trackercap cv2.VideoCapture(0) if not cap.isOpened():print("无法打开 RTSP 流&#xff0c;请检查地址、网络连接或 GStreamer 配置。") else:# 创建共…

网页常见水印实现方式

文章目录 1 明水印技术实现1.1 DOM覆盖方案1.2 Canvas动态渲染1.3 CSS伪元素方案2 暗水印技术解析2.1 空域LSB算法2.2 频域傅里叶变换3 防篡改机制设计3.1 MutationObserver防护3.2 Canvas指纹追踪4 前后端实现对比5 攻防博弈深度分析5.1 常见破解手段5.2 进阶防御策略6 选型近…

【ARM】MDK如何将变量存储到指定内存地址

1、 文档目标 在嵌入式系统开发中&#xff0c;通过MDK&#xff08;Microcontroller Development Kit&#xff09;进行工程配置&#xff0c;将指定的变量存储到指定的内存地址上是一项非常重要的技术。这项操作不仅能够满足特定硬件架构的需求&#xff0c;还能优化系统的性能和…