亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

C++深入刨析muduo中的抽象類Poller

 更新時間:2022年04月24日 16:12:29   作者:BugMaker-shen  
muduo網(wǎng)絡(luò)庫中Poller類是一個抽象類,用戶使用PollPoller或者EPollPoller類,下面跟隨小編一起來詳細(xì)了解一下

Poller是抽象類,Eventloop通過抽象類Poller,引用不同的派生類對象(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ù),是抽象類

為什么muduo庫要抽象一層Poller呢?

因為在eventloop里面,在使用I/O復(fù)用的時候,并沒有直接指定epoll,因為muduo庫對外提供兩種I/O復(fù)用方法poll和epoll,在eventloop里面,沒有直接使用poll或者epoll,而是從抽象層面通過抽象類Poller,引用不同的派生類對象,調(diào)用同名覆蓋方法,就可以很方便地去擴(kuò)展不同的I/O復(fù)用

Poller抽象基類有兩個成員變量:

 protected:
  typedef std::map<int, Channel*> ChannelMap;
  ChannelMap channels_;
 private:
  EventLoop* ownerLoop_;  // 表示Poller所屬的事件循環(huán)EventLoop

Poller監(jiān)聽的就是Eventloop另外一個成員ChannelList保存的那些channel,所以Poller里面會有一個ChannelMap,key是sockfd,value是sockfd所屬的Channel

protected的成員變量就是讓派生類可以訪問到,private的成員變量派生類不能訪問到

Poller.h

#pragma once
#include "noncopyable.h"
#include "Timestamp.h"
#include <vector>
#include <unordered_map>
class Channel;  //只用到指針類型,如果需要用到實例就要包含相應(yīng)頭文件,類型聲明是沒用的
class EventLoop;
// muduo庫中多路事件分發(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里是否包含某個channel
        bool hasChannel(Channel* channel) const;
        // EventLoop可以通過newDefaultPoller獲取一個Poller實例
        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的實現(xiàn)放在Poller.cc

如果真的把newDefaultPoller寫在Poller.cc里面,從語法上來說,沒有錯誤。但是這個函數(shù)是要生成一個具體的I/O復(fù)用對象,并返回一個基類的指針

所以基類就得include這兩個包含了派生類聲明的頭文件,才能去生成一個具體的實例對象并返回回去,這樣不合理。

在繼承結(jié)構(gòu)中,Poller是基類,只能派生類引用基類,而Poller.cc基類不能引用派生類,這就是好的OOP設(shè)計

muduo用一個單獨的源文件DefaultPoller.cc實現(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實例
  }
  else
  {
    return new EPollPoller(loop); // 默認(rèn)返回EPollPoller實例
  }
}

重寫DefaultPoller.cc

#include <stdlib.h>
#include "Poller.h"
#include "EPollPoller.h"
Poller* Poller::newDefaultPoller(EventLoop* loop){
    if(std::getenv("MUDUO_USE_POLL")){
        // new poll的實例
        return nullptr;
    }else{
        return new EPollPoller(loop);
    }
}

到此這篇關(guān)于C++深入刨析muduo中的抽象類Poller的文章就介紹到這了,更多相關(guān)C++ 抽象類Poller內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:

相關(guān)文章

  • C語言實現(xiàn)一個閃爍的圣誕樹

    C語言實現(xiàn)一個閃爍的圣誕樹

    本文詳細(xì)講解了C語言實現(xiàn)一個閃爍的圣誕樹,文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-12-12
  • C++設(shè)計模式編程中的觀察者模式使用示例

    C++設(shè)計模式編程中的觀察者模式使用示例

    這篇文章主要介紹了C++設(shè)計模式編程中的觀察者模式使用示例,觀察者模式在被觀察者和觀察者之間建立一個抽象的耦合,需要的朋友可以參考下
    2016-03-03
  • C++ 強(qiáng)制類型轉(zhuǎn)換詳解

    C++ 強(qiáng)制類型轉(zhuǎn)換詳解

    這篇文章主要介紹的是C++ 強(qiáng)制類型轉(zhuǎn)換詳解,C語言中的強(qiáng)制轉(zhuǎn)換主要用于普通數(shù)據(jù)類型、指針的強(qiáng)制轉(zhuǎn)換,沒有類型檢查,轉(zhuǎn)換不安全,下面我們來看看其具體語法及詳細(xì)內(nèi)容
    2021-11-11
  • 深入了解C++ 結(jié)構(gòu)體(struct)與共用體(union)

    深入了解C++ 結(jié)構(gòu)體(struct)與共用體(union)

    這篇文章主要介紹了C++ 結(jié)構(gòu)體與共用體的的相關(guān)資料,幫助大家更好的理解和學(xué)習(xí)c++,感興趣的朋友可以了解下
    2020-08-08
  • c++特殊構(gòu)造函數(shù)詳解

    c++特殊構(gòu)造函數(shù)詳解

    大家好,本篇文章主要講的是c++特殊構(gòu)造函數(shù)詳解,感興趣的同學(xué)趕快來看一看吧,對你有幫助的話記得收藏一下,方便下次瀏覽
    2022-01-01
  • C++中的vector使用詳解及重要部分底層實現(xiàn)

    C++中的vector使用詳解及重要部分底層實現(xiàn)

    本篇文章會對vector的語法使用進(jìn)行詳解,同時,還會對重要難點部分的底層實現(xiàn)進(jìn)行講解,其中有vector的迭代器失效和深拷貝問題,希望本篇文章的內(nèi)容會對你有所幫助
    2023-07-07
  • C++中圖片重命名實現(xiàn)代碼

    C++中圖片重命名實現(xiàn)代碼

    這篇文章主要介紹了C++中圖片重命名實現(xiàn)代碼,需要的朋友可以參考下
    2017-05-05
  • C/C++實現(xiàn)對STORM運(yùn)行信息查看及控制的方法

    C/C++實現(xiàn)對STORM運(yùn)行信息查看及控制的方法

    這篇文章主要介紹了C/C++實現(xiàn)對STORM運(yùn)行信息查看及控制的方法,需要的朋友可以參考下
    2014-07-07
  • OpenCV實現(xiàn)幀差法檢測運(yùn)動目標(biāo)

    OpenCV實現(xiàn)幀差法檢測運(yùn)動目標(biāo)

    這篇文章主要為大家詳細(xì)介紹了OpenCV實現(xiàn)幀差法檢測運(yùn)動目標(biāo),文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-03-03
  • C/C++實現(xiàn)快速排序算法的兩種方式實例

    C/C++實現(xiàn)快速排序算法的兩種方式實例

    快速排序是一種采用分治思想,在實踐中通常運(yùn)行較快一種排序算法,這篇文章主要給大家介紹了關(guān)于C/C++實現(xiàn)快速排序的兩種方式的相關(guān)資料,文中給出了詳細(xì)的示例代碼,需要的朋友可以參考下
    2021-08-08

最新評論