C++深入刨析muduo中的抽象類(lèi)Poller
Poller是抽象類(lèi),Eventloop通過(guò)抽象類(lèi)Poller,引用不同的派生類(lèi)對(duì)象(PollPoller或EpollPoller),調(diào)用同名覆蓋方法,就可以很方便地去擴(kuò)展不同的I/O復(fù)用
Poller.h源碼
#include <map>
#include <vector>
#include "muduo/base/Timestamp.h"
#include "muduo/net/EventLoop.h"
namespace muduo
{
namespace net
{
class Channel;
class Poller : noncopyable
{
public:
typedef std::vector<Channel*> ChannelList;
Poller(EventLoop* loop);
virtual ~Poller();
virtual Timestamp poll(int timeoutMs, ChannelList* activeChannels) = 0;
virtual void updateChannel(Channel* channel) = 0;
virtual void removeChannel(Channel* channel) = 0;
virtual bool hasChannel(Channel* channel) const;
static Poller* newDefaultPoller(EventLoop* loop);
void assertInLoopThread() const
{
ownerLoop_->assertInLoopThread();
}
protected:
typedef std::map<int, Channel*> ChannelMap;
ChannelMap channels_;
private:
EventLoop* ownerLoop_;
};
} // namespace net
} // namespace muduo可以看到,Poller里有很多純虛函數(shù),是抽象類(lèi)
為什么muduo庫(kù)要抽象一層Poller呢?
因?yàn)樵趀ventloop里面,在使用I/O復(fù)用的時(shí)候,并沒(méi)有直接指定epoll,因?yàn)閙uduo庫(kù)對(duì)外提供兩種I/O復(fù)用方法poll和epoll,在eventloop里面,沒(méi)有直接使用poll或者epoll,而是從抽象層面通過(guò)抽象類(lèi)Poller,引用不同的派生類(lèi)對(duì)象,調(diào)用同名覆蓋方法,就可以很方便地去擴(kuò)展不同的I/O復(fù)用
Poller抽象基類(lèi)有兩個(gè)成員變量:
protected: typedef std::map<int, Channel*> ChannelMap; ChannelMap channels_; private: EventLoop* ownerLoop_; // 表示Poller所屬的事件循環(huán)EventLoop
Poller監(jiān)聽(tīng)的就是Eventloop另外一個(gè)成員ChannelList保存的那些channel,所以Poller里面會(huì)有一個(gè)ChannelMap,key是sockfd,value是sockfd所屬的Channel
protected的成員變量就是讓派生類(lèi)可以訪問(wèn)到,private的成員變量派生類(lèi)不能訪問(wèn)到
Poller.h
#pragma once
#include "noncopyable.h"
#include "Timestamp.h"
#include <vector>
#include <unordered_map>
class Channel; //只用到指針類(lèi)型,如果需要用到實(shí)例就要包含相應(yīng)頭文件,類(lèi)型聲明是沒(méi)用的
class EventLoop;
// muduo庫(kù)中多路事件分發(fā)器的核心IO復(fù)用模塊
class Poller : noncopyable{
public:
using ChannelList = std::vector<Channel*>;
Poller(EventLoop *loop);
virtual ~Poller();
// 給所有IO復(fù)用保留統(tǒng)一的接口 epoll_wait
virtual Timestamp poll(int timeoutMs, ChannelList* activeChannels) = 0;
// 更新感興趣的事件 epoll_ctl EPOLL_CTL_ADD EPOLL_CTL_MOD
virtual void updateChannel(Channel* channel) = 0;
// eventloop中刪除channel epoll_ctl EPOLL_CTL_DEL
virtual void removeChannel(Channel* channel) = 0;
// 判斷Poller里是否包含某個(gè)channel
bool hasChannel(Channel* channel) const;
// EventLoop可以通過(guò)newDefaultPoller獲取一個(gè)Poller實(shí)例
static Poller* newDefaultPoller(EventLoop* loop);
protected:
// key:sockfd,value:sockfd所屬的Channel
using ChannelMap = std::unordered_map<int, Channel*>;
ChannelMap channels_;
private:
EventLoop* ownerLoop_; // Poller所屬的事件循環(huán)
};Poller.cc
#include "Poller.h"
#include "Channel.h"
Poller::Poller(EventLoop *loop)
: ownerLoop_(loop)
{}
Poller::~Poller() = default;
bool Poller::hasChannel(Channel* channel) const{
auto iter = channels_.find(channel->fd());
return iter != channels_.end() && iter->second == channel; // 找到fd且channel相等
}為什么不把newDefaultPoller的實(shí)現(xiàn)放在Poller.cc里

如果真的把newDefaultPoller寫(xiě)在Poller.cc里面,從語(yǔ)法上來(lái)說(shuō),沒(méi)有錯(cuò)誤。但是這個(gè)函數(shù)是要生成一個(gè)具體的I/O復(fù)用對(duì)象,并返回一個(gè)基類(lèi)的指針
所以基類(lèi)就得include這兩個(gè)包含了派生類(lèi)聲明的頭文件,才能去生成一個(gè)具體的實(shí)例對(duì)象并返回回去,這樣不合理。
在繼承結(jié)構(gòu)中,Poller是基類(lèi),只能派生類(lèi)引用基類(lèi),而Poller.cc基類(lèi)不能引用派生類(lèi),這就是好的OOP設(shè)計(jì)
muduo用一個(gè)單獨(dú)的源文件DefaultPoller.cc實(shí)現(xiàn)newDefaultPoller
#include "muduo/net/Poller.h"
#include "muduo/net/poller/PollPoller.h"
#include "muduo/net/poller/EPollPoller.h"
#include <stdlib.h>
using namespace muduo::net;
Poller* Poller::newDefaultPoller(EventLoop* loop)
{
if (::getenv("MUDUO_USE_POLL"))
{
return new PollPoller(loop); // 環(huán)境變量中有MUDUO_USE_POLL,則返回PollPoller實(shí)例
}
else
{
return new EPollPoller(loop); // 默認(rèn)返回EPollPoller實(shí)例
}
}重寫(xiě)DefaultPoller.cc
#include <stdlib.h>
#include "Poller.h"
#include "EPollPoller.h"
Poller* Poller::newDefaultPoller(EventLoop* loop){
if(std::getenv("MUDUO_USE_POLL")){
// new poll的實(shí)例
return nullptr;
}else{
return new EPollPoller(loop);
}
}
到此這篇關(guān)于C++深入刨析muduo中的抽象類(lèi)Poller的文章就介紹到這了,更多相關(guān)C++ 抽象類(lèi)Poller內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C語(yǔ)言實(shí)現(xiàn)一個(gè)閃爍的圣誕樹(shù)
本文詳細(xì)講解了C語(yǔ)言實(shí)現(xiàn)一個(gè)閃爍的圣誕樹(shù),文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-12-12
C++ 強(qiáng)制類(lèi)型轉(zhuǎn)換詳解
這篇文章主要介紹的是C++ 強(qiáng)制類(lèi)型轉(zhuǎn)換詳解,C語(yǔ)言中的強(qiáng)制轉(zhuǎn)換主要用于普通數(shù)據(jù)類(lèi)型、指針的強(qiáng)制轉(zhuǎn)換,沒(méi)有類(lèi)型檢查,轉(zhuǎn)換不安全,下面我們來(lái)看看其具體語(yǔ)法及詳細(xì)內(nèi)容2021-11-11
深入了解C++ 結(jié)構(gòu)體(struct)與共用體(union)
這篇文章主要介紹了C++ 結(jié)構(gòu)體與共用體的的相關(guān)資料,幫助大家更好的理解和學(xué)習(xí)c++,感興趣的朋友可以了解下2020-08-08
C++中的vector使用詳解及重要部分底層實(shí)現(xiàn)
本篇文章會(huì)對(duì)vector的語(yǔ)法使用進(jìn)行詳解,同時(shí),還會(huì)對(duì)重要難點(diǎn)部分的底層實(shí)現(xiàn)進(jìn)行講解,其中有vector的迭代器失效和深拷貝問(wèn)題,希望本篇文章的內(nèi)容會(huì)對(duì)你有所幫助2023-07-07
C/C++實(shí)現(xiàn)對(duì)STORM運(yùn)行信息查看及控制的方法
這篇文章主要介紹了C/C++實(shí)現(xiàn)對(duì)STORM運(yùn)行信息查看及控制的方法,需要的朋友可以參考下2014-07-07
OpenCV實(shí)現(xiàn)幀差法檢測(cè)運(yùn)動(dòng)目標(biāo)
這篇文章主要為大家詳細(xì)介紹了OpenCV實(shí)現(xiàn)幀差法檢測(cè)運(yùn)動(dòng)目標(biāo),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-03-03
C/C++實(shí)現(xiàn)快速排序算法的兩種方式實(shí)例
快速排序是一種采用分治思想,在實(shí)踐中通常運(yùn)行較快一種排序算法,這篇文章主要給大家介紹了關(guān)于C/C++實(shí)現(xiàn)快速排序的兩種方式的相關(guān)資料,文中給出了詳細(xì)的示例代碼,需要的朋友可以參考下2021-08-08

