muduo源码分析--事件回调层次是怎么传递的Tcpserver Channel TcpConnection

本文深入分析muduo库的事件回调机制,从EventLoop的epoll_wait开始,探讨如何将激活的事件传递到Channel,并通过Channel的handleEvent函数调用相应的回调函数。接着,文章揭示TcpConnection如何设置并调用writeCallback_、readCallback_和closeCallback_。最后,说明TcpServer如何将用户函数注册到TcpConnection,完成事件处理的完整流程。

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

muduo库中的源码并不是很多,但是回调的处理非常巧妙,这里从事件激活(某个套接字上可读/可写)以后这个层次看回调怎么被调用的。
首先从最大的EventLoop说起:
        EventLoop中拥有事件链表(每一个元素都是Channel),在loop函数中调用epoll_wait系统调用的时候,将EpollPollr中EventList,将这个链表中激活事件添加到EventLoop中的activeChannel中,
所以暂时不管怎么弄EventLoop中的loop()函数已经将所有祖册到这个EventLoop上的被激活fd添加到了其私有变量activeChannel中了(助于怎么添加稍后再议),得到激活链表,对每个Channel调用其handleEvent,这样就结束了这次poll的循环。

        接下来议Channel中的handleEvent函数
Channel:
        Channel并不拥有fd,在这里有三个回调函数,writeCallback_ readCallback_ closeCallback_ 
        在handleEvent函数中,根据事件类型执行相应的回调,就是上面的三个回调函数,那么这三个回调函数怎么来的呢?肯定是谁拥有Channel谁就会Channel中的这个三个回调赋值。用内有channel的类有Acceptor Connector EventLoop TcpConn
### 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]。 ---
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值