热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

muduo源码分析之EventLoop、Channel、Poller的实现

作者一直强调的一个概念叫做oneloopperthread,撇开多线程不谈,本篇博文将学习,怎么将传统的IO复用pollepoll封装到C++类中。1.IO复用复习使用p

作者一直强调的一个概念叫做one loop per thread,撇开多线程不谈,本篇博文将学习,怎么将传统的I/O复用poll/epoll封装到C++ 类中。

1.I/O复用

复习使用poll/epoll进行I/O复用的一些编程内容。

使用poll

对于一个文件描述符fd来说,我们将通过struct pollfd来设置我们关注的事件event,并在通过poll调用返回获取活跃的事件revent。比如说(伪代码):

struct pollfd pfds[2];
fd1 = GET_FD();
fd2 = GET_FD();

pfds[0].fd=fd1;
pfds[0].events=POLLIN;
//fd2 is the same
while(true)
{
ret=poll(pfds,pfds中关心的描述符数目,TIMEOUT);
if(pfds[0].revents & POLLOUT)
{
///
}
///
}

使用epoll

epoll就要复杂一些了,主要由epoll_createepoll_ctlepoll_wait函数组成,比如说(伪代码):

epollfd=epoll_create()
struct epoll_event ev;
ev.event=EPOLL_IN;
ev.data.fd=fd;
epoll_ctl(epollfd,EPOLL_CTL_ADD,fd,&ev);
while(true)
{
epoll_wait();
handle_event();
}

2.什么都不做的EventLoop

本文代码参见https://github.com/chenshuo/recipes

one loop per thread表示每个线程只能有一个EventLoop对象,怎么来保证呢?
作者用__thread关键字修饰的t_loopInThisThread来存储对象指针(它的值在多线程环境下互不干扰),每当EventLoop进行构造的时候,判断该指针是否为NULL也就是在之前有没有被初始化过。

// reactor/s00/EventLoop.cc

/*__thread关键字 各个线程之间的值相互不干扰*/
__thread EventLoop* t_loopInThisThread = 0;

EventLoop::EventLoop()
: looping_(false),
threadId_(CurrentThread::tid())
{
LOG_TRACE <<"EventLoop created " <<this <<" in thread " < /*t_loopInThisThread不为空表示当前线程创建了EventLoop对象*/
/*one thread per loop表示一个线程只能有一个EventLoop对象*/
if (t_loopInThisThread)
{

LOG_FATAL <<"Another EventLoop " < <<" exists in this thread " < }
else
{
/*记录该线程的EventLoop对象*/
t_loopInThisThread = this;
}
}

对于一个简单的EventLoop来说,poll调用封装在EventLoop::loop()中,比如说:

void EventLoop::loop()
{
assert(!looping_);
/*确保EventLoop在创建它的线程中loop调用*/
assertInLoopThread();
/*等待5秒的poll调用*/
::poll(NULL, 0, 5*1000);
LOG_TRACE <<"EventLoop " <<this <<" stop looping";

}

3.Reactor关键结构

主要介绍Channel类(用于对描述符及事件的封装和事件的分发)Poller类(主要对poll / epoll调用的封装,但只是获取活跃事件并不负责事件分发)

我一直思考的问题就在于ChannelPollerEventLoop是如何协调工作的,所以,先从一个示例看起。

// reactor/s01/test3.cc

muduo::EventLoop* g_loop;

void timeout()
{
printf("Timeout!\n");
g_loop->quit();
}

int main()
{
muduo::EventLoop loop;//EventLoop对象
g_loop = &loop;


//关心的描述符
int timerfd = ::timerfd_create(CLOCK_MONOTONIC, TFD_NONBLOCK | TFD_CLOEXEC);
//创建channel对象,保存一份EventLoop的指针
muduo::Channel channel(&loop, timerfd);
//设置描述符可读时执行的回调函数
channel.setReadCallback(timeout);
//该函数用于设置关心的事件event
//并且会依次调用Channel::update ---> EventLoop::updateChannel --->Poller::updateChannel
channel.enableReading();

struct itimerspec howlong;
bzero(&howlong, sizeof howlong);
howlong.it_value.tv_sec = 5;
::timerfd_settime(timerfd, 0, &howlong, NULL);

//loop循环
loop.loop();

::close(timerfd);
}

程序执行的结果是5秒之后timefd可读将执行回调函数,并关闭loop。

现在我们来阅读相关源码,从上面的注释,首先,需要构造一个Channel对象:

Channel::Channel(EventLoop* loop, int fdArg)
: loop_(loop),//保存EventLoop对象的指针
fd_(fdArg),//描述符
events_(0),//关心的事件
revents_(0),//活跃的事件
index_(-1)//index_用于记录当前fd在pollfd数组中的位置
{
}

之后设置回调,这里用到boost::function和boost::bind就不再详细描述。

enableReading干了两件事情:
1、设置关心的事情为POLL_IN,事件可读。
2、将fd添加到pollfd数组中。
第2步将依次调用Channel::update ---> EventLoop::updateChannel --->Poller::updateChannel
分别来看看这些函数:

void Channel::update()
{
loop_->updateChannel(this);//这里将channel对象的指针传到了EventLoop里面
/*loop->updateChannel--->Poller::updateChannel*/
}
/////////////////////////////////////////////////////////////////////////////
void EventLoop::updateChannel(Channel* channel)
{
assert(channel->ownerLoop() == this);
assertInLoopThread();
//poller_是EventLoop中Poller类型的成员通过EventLoop的构造函数初始化。
poller_->updateChannel(channel);//将channel对象的指针传给了Poller对象
}
/////////////////////////////////////////////////////////////////////////
void Poller::updateChannel(Channel* channel)
{
/*与其ownerLoop在同一个线程*/
assertInLoopThread();
LOG_TRACE <<"fd = " <fd() <<" events = " <events();
if (channel->index() <0) {//添加时Channel::index为初始化值-1
/*添加一个新的*/
/*每次都在尾部添加*/
// a new one, add to pollfds_
assert(channels_.find(channel->fd()) == channels_.end());
struct pollfd pfd;//pollfd结构体
//进行设置
pfd.fd = channel->fd();
pfd.events = static_cast<short>(channel->events());
pfd.revents = 0;
//pollfds_是Poller类中的成员是一个pollfd类型的 vector,充当poll调用的第一个参数
pollfds_.push_back(pfd);
int idx = static_cast<int>(pollfds_.size())-1;//重新计算索引,方便下次更新时快速定位
channel->set_index(idx);//类似于设置上下文
channels_[pfd.fd] = channel;//channels_是fd和Channel的一个map
} else {
// update existing one
// 更新fd,由于channel保存有index,因此访问的时间效率为O(1)
// 可能修改event后进入该分分支。
assert(channels_.find(channel->fd()) != channels_.end());
assert(channels_[channel->fd()] == channel);
int idx = channel->index();
assert(0 <= idx && idx <static_cast<int>(pollfds_.size()));
struct pollfd& pfd = pollfds_[idx];
assert(pfd.fd == channel->fd() || pfd.fd == -1);
pfd.events = static_cast<short>(channel->events());
pfd.revents = 0;
if (channel->isNoneEvent()) {
// ignore this pollfd
pfd.fd = -1;
}
}
}

到目前位置,我们已经将我们关心的描述符及事件都设置好了,现在就需要进行真正的poll调用了,也就是loop循环

void EventLoop::loop()
{
assert(!looping_);
assertInLoopThread();
looping_ = true;
quit_ = false;

while (!quit_)
{
activeChannels_.clear();
//在这个poll调用中封装了真正的poll(2)
poller_->poll(kPollTimeMs, &activeChannels_);
for (ChannelList::iterator it = activeChannels_.begin();
it != activeChannels_.end(); ++it)
{
(*it)->handleEvent();
}
}

LOG_TRACE <<"EventLoop " <<this <<" stop looping";
looping_ = false;
}

EventLoop类中保存了一个vector类型的activeChannels,用来获取活跃的事件,因为从上文Poller类中的一个channel变量(map)就可以看出Channel和fd是可以一一对应的。

再来看看Poller封装的poll调用。

Timestamp Poller::poll(int timeoutMs, ChannelList* activeChannels)
{
// XXX pollfds_ shouldn't change
int numEvents = ::poll(&*pollfds_.begin(), pollfds_.size(), timeoutMs);//poll调用
LOG_INFO< Timestamp now(Timestamp::now());
if (numEvents > 0) {
LOG_TRACE <" events happended";
/*填写活动事件*/
fillActiveChannels(numEvents, activeChannels);//将活跃的channel添加到avtiveChannel中去
//并且设置本channel对象的revent
} else if (numEvents == 0) {
LOG_TRACE <<" nothing happended";
} else {
LOG_INFO<<"time out";
LOG_SYSERR <<"Poller::poll()";
}
return now;
}

最后,看到一个for循环,因为活跃的描述符/Channel可能很多,muduo 的做法,是现将所有活跃的描述符/Channel用activeChannel管理起来再迭代一次,对于每个Channel调用它们的handleEvent方法,该方法就根据设置的revent调用之前设置的回调方法。

void Channel::handleEvent()
{
/*活动事件回调*/
if (revents_ & POLLNVAL) {
LOG_WARN <<"Channel::handle_event() POLLNVAL";
}

if (revents_ & (POLLERR | POLLNVAL)) {
if (errorCallback_) errorCallback_();
}
if (revents_ & (POLLIN | POLLPRI | POLLRDHUP)) {
if (readCallback_) readCallback_();
}
if (revents_ & POLLOUT) {
if (writeCallback_) writeCallback_();
}
}

4.参考

1.Linux多线程服务端编程 使用MuduoC++网络库
2.http://blog.csdn.net/jnu_simba/article/details/14486661


推荐阅读
  • 在高并发需求的C++项目中,我们最初选择了JsonCpp进行JSON解析和序列化。然而,在处理大数据量时,JsonCpp频繁抛出异常,尤其是在多线程环境下问题更为突出。通过分析发现,旧版本的JsonCpp存在多线程安全性和性能瓶颈。经过评估,我们最终选择了RapidJSON作为替代方案,并实现了显著的性能提升。 ... [详细]
  • 深入解析 Apache Shiro 安全框架架构
    本文详细介绍了 Apache Shiro,一个强大且灵活的开源安全框架。Shiro 专注于简化身份验证、授权、会话管理和加密等复杂的安全操作,使开发者能够更轻松地保护应用程序。其核心目标是提供易于使用和理解的API,同时确保高度的安全性和灵活性。 ... [详细]
  • Java多线程实现:从1到100分段求和并汇总结果
    本文介绍如何使用Java编写一个程序,通过10个线程分别计算不同区间的和,并最终汇总所有线程的结果。每个线程负责计算一段连续的整数之和,最后将所有线程的结果相加。 ... [详细]
  • 并发编程:深入理解设计原理与优化
    本文探讨了并发编程中的关键设计原则,特别是Java内存模型(JMM)的happens-before规则及其对多线程编程的影响。文章详细介绍了DCL双重检查锁定模式的问题及解决方案,并总结了不同处理器和内存模型之间的关系,旨在为程序员提供更深入的理解和最佳实践。 ... [详细]
  • 本文详细探讨了JDBC(Java数据库连接)的内部机制,重点分析其作为服务提供者接口(SPI)框架的应用。通过类图和代码示例,展示了JDBC如何注册驱动程序、建立数据库连接以及执行SQL查询的过程。 ... [详细]
  • 深入探讨CPU虚拟化与KVM内存管理
    本文详细介绍了现代服务器架构中的CPU虚拟化技术,包括SMP、NUMA和MPP三种多处理器结构,并深入探讨了KVM的内存虚拟化机制。通过对比不同架构的特点和应用场景,帮助读者理解如何选择最适合的架构以优化性能。 ... [详细]
  • 实体映射最强工具类:MapStruct真香 ... [详细]
  • dotnet 通过 Elmish.WPF 使用 F# 编写 WPF 应用
    本文来安利大家一个有趣而且强大的库,通过F#和C#混合编程编写WPF应用,可以在WPF中使用到F#强大的数据处理能力在GitHub上完全开源Elmis ... [详细]
  • 微软Exchange服务器遭遇2022年版“千年虫”漏洞
    微软Exchange服务器在新年伊始遭遇了一个类似于‘千年虫’的日期处理漏洞,导致邮件传输受阻。该问题主要影响配置了FIP-FS恶意软件引擎的Exchange 2016和2019版本。 ... [详细]
  • FinOps 与 Serverless 的结合:破解云成本难题
    本文探讨了如何通过 FinOps 实践优化 Serverless 应用的成本管理,提出了首个 Serverless 函数总成本估计模型,并分享了多种有效的成本优化策略。 ... [详细]
  • 深入解析Java虚拟机(JVM)架构与原理
    本文旨在为读者提供对Java虚拟机(JVM)的全面理解,涵盖其主要组成部分、工作原理及其在不同平台上的实现。通过详细探讨JVM的结构和内部机制,帮助开发者更好地掌握Java编程的核心技术。 ... [详细]
  • 深入解析 Android IPC 中的 Messenger 机制
    本文详细介绍了 Android 中基于消息传递的进程间通信(IPC)机制——Messenger。通过实例和源码分析,帮助开发者更好地理解和使用这一高效的通信工具。 ... [详细]
  • 深入解析Java多线程与并发库的应用:空中网实习生面试题详解
    本文详细探讨了Java多线程与并发库的高级应用,结合空中网在挑选实习生时的面试题目,深入分析了相关技术要点和实现细节。文章通过具体的代码示例展示了如何使用Semaphore和SynchronousQueue来管理线程同步和任务调度。 ... [详细]
  • 本文作者分享了在阿里巴巴获得实习offer的经历,包括五轮面试的详细内容和经验总结。其中四轮为技术面试,一轮为HR面试,涵盖了大量的Java技术和项目实践经验。 ... [详细]
  • 本文介绍了如何在多线程环境中实现异步任务的事务控制,确保任务执行的一致性和可靠性。通过使用计数器和异常标记字段,系统能够准确判断所有异步线程的执行结果,并根据结果决定是否回滚或提交事务。 ... [详细]
author-avatar
开发者小白
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有