muduo库的TcpConnection类剖析

本文详细剖析了muduo库中的TcpConnection类,它作为已连接套接字的抽象,利用Channel处理IO事件。在构造时,TcpConnection注册handleRead到ReadCallback,并在readable事件中调用MessageCallback。CloseCallback用于通知TcpServer和TcpClient移除TcpConnection。在newConnection中,TcpServer将removeConnection注册到TcpConnection的setCloseCallback。当连接断开,TcpConnection回调removeConnection并从Map中移除。此外,还探讨了TcpConnection的生命周期管理和事件处理机制。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

本文链接:https://blog.youkuaiyun.com/Mr_scott_j/article/details/111059068

TcpConnection是muduo中最为复杂的类,其实就是对已连接套接字的一个抽象。

TcpConnection使用Channel来获得socket上的IO事件,它可以自己处理writable事件,而把readable事件通过MessageCallback传达给客户。在TcpConnection析构时候会close(fd)(在Socket析构函数中发生)。

TcpConnection在构造函数中将handleRead注册到Channel::ReadCallback上,并在其中调用messageCallback_;而handleWrite由自己处理。

TcpConnection::TcpConnection(EventLoop* loop,
                             const string& nameArg,
                             int sockfd,
                             const InetAddress& localAddr,
                             const InetAddress& peerAddr)
  : loop_(CHECK_NOTNULL(loop)),//不能为空,否则触发FATAL
    name_(nameArg),
    state_(kConnecting),
    reading_(true),
    socket_(new Socket(sockfd)),
    channel_(new Channel(loop, sockfd)),
    localAddr_(localAddr),//本地地址
    peerAddr_(peerAddr),
    highWaterMark_(64*1024*1024)
{
   
   
//handleRead中会调用messageCallback_
  channel_->setReadCallback(
      std::bind(&TcpConnection::handleRead, this, _1));
  channel_->setWriteCallback(
      std::bind
### muduo::net::TcpConnection 的用法与 SSL 连接实现分析 #### 1. **muduo::net::TcpConnection 概述** `muduo::net::TcpConnection` 是一个封装了 TCP 连接的对象,主要用于处理客户端和服务端之间的通信。该提供了多种方法来管理连接状态、发送和接收数据等功能[^3]。然而,原生的 `muduo` 并未内置对 SSL/TLS 支持的功能,因此如果需要使用 SSL 加密传输,则需自行扩展或集成第三方(如 OpenSSL)。 #### 2. **TcpConnection 的核心成员变量与方法** 以下是 `TcpConnection` 的几个重要组成部分及其功能描述: - **`std::string name_`**: 表示连接的名字,通常由用户自定义或者系统生成唯一标识符。 - **`Socket socket_`**: 封装底层文件描述符及相关操作接口。 - **`Channel channel_`**: 绑定到指定事件循环中监控 I/O 活动变化的通知机制。 - **`Buffer inputBuffer_, outputBuffer_`**: 缓存来自远端的数据包以及待发送的消息队列内容存储区域。 对于常规非加密模式下工作而言,可以通过如下方式创建并启动一条新的 TCP 链路实例[^3]: ```cpp // 创建一个新的 TcpConnection 对象 TcpConnectionPtr conn(new TcpConnection(loop, sock, localAddr, peerAddr)); conn->setConnectionCallback(onConnection); // 设置连接回调函数 conn->setMessageCallback(onMessage); // 设置消息到达时触发的动作处理器 conn->setWriteCompleteCallback(onWriteComplete); // 数据写入完成后执行的任务指针 conn->connectEstablished(); // 明确告知已成功握手完毕可以正常收发信息了 ``` #### 3. **基于 OpenSSL 扩展支持 SSL 功能** 为了使得现有的基础框架能够满足更高层次的安全防护需求,我们可以引入开源工具集 OpenSSL 来达成目标效果。具体做法如下所示: ##### (1)**初始化阶段准备** 在应用程序启动初期加载必要组件,并准备好相应的配置项供后续环节调用: ```cpp SSL_CTX* InitCTX(void){ const SSL_METHOD *method; SSL_CTX *ctx; OpenSSL_add_all_algorithms(); /* Load cryptos, et.al. */ SSL_load_error_strings(); /* Bring in and register error messages */ method = TLS_server_method(); /* Create server instance */ ctx = SSL_CTX_new(method); if (ctx == NULL){ ERR_print_errors_fp(stderr); abort(); } return ctx; } ``` ##### (2)**改造原有代码路径加入新特性** 针对每一次新建成功的普通套接字对象,在正式投入使用前先将其转换成具备加解密能力的新形态实体后再交付给高层模块继续运作下去: ```cpp class SslTcpConnection : public muduo::net::TcpConnection{ public: explicit SslTcpConnection(EventLoop* loop, const std::string& nameArg, int sockfd, const InetAddress& localAddr, const InetAddress& peerAddr) : muduo::net::TcpConnection(loop,nameArg,sockfd,localAddr,peerAddr), ssl_(nullptr),sslCtx_(InitCTX()){ setupSsl(); } private: void setupSsl(){ ssl_=SSL_new(sslCtx_); BIO *bio=BIO_new_socket(fd(),BIO_NOCLOSE); SSL_set_bio(ssl_, bio,bio); if(SSL_accept(ssl_)<=0){ handleError("Error during SSL accept"); } } void send(const void* message,size_t len) override { char buf[BUFLEN]; size_t bytes=SSL_write(ssl_,message,len); if(bytes<0){ handleError("Failed writing data over SSL connection."); } } ssize_t read(void* buffer,size_t maxBytes)override{ return SSL_read(ssl_,buffer,maxBytes); } ~SslTcpConnection(){ SSL_free(ssl_); SSL_CTX_free(sslCtx_); } protected: SSL* ssl_; SSL_CTX* sslCtx_; }; ``` 以上即完成了从标准版升级至增强型版本的主要改动部分介绍[^1]。 ---
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值