仿Muduo库实现高并发服务器——TcpServer模块
创始人
2024-09-26 06:47:47
0

TcpServer模块的公开接口:

        对于TcpServer对象设置回调函数,其实最终会被设置到Connection对象上,而每个客户端对于一个Connection对象。 

        启动非活跃连接销毁,Tcpserver对象默认是关闭非活跃连接销毁的,添加定时任务下面会讲。

         TcpServer对象创建后,就需要调用Start()函数进行线程池的创建,以及主线程EventLooop对象的运行,这个就是死循环。

添加定时任务:

         这个添加定时任务是添加到主线程中的EventLoop对象的定时器对象当中去的。

         而子线程是不会添加这种定时器任务,他只会运行非活跃连接的销毁工作。

连接的删除:

当连接断开时,需要将断开的连接从容器中删除。

        这是设置给新连接的关闭回调函数,注意这个需要最后删除,否则Connection对象中保存的资源就不存在了,如果再去访问这些资源会出错,会什么还要去访问这些资源,因为如果你先删除Connection对象,就会找不到那些还没删除的资源,造成资源泄漏。

 

创建新连接并且为新连接设置回调函数:

        新连接的回调函数是外部进行设置的。

 上面就是有新连接到来调用的回调函数。

        为新连接分配EventLoop对象和_next_id,将回调函数设置给Connection对象,在connection对象连接的不同状态中,进行调用。这不是设置给Channel对象的,channel对象设置的回调函数,是Connection对象的私有成员函数

        该回调函数最终会被设置给Acceptor对象。

 

        当有新连接到来时,就会用新连接的文件描述符,初始化Connection,并将初始化好的connection对象放到存储容器中。 

TcpServer模块中的成员变量:

_next_id:

        他是自动增长的,可以标记连接的唯一性。

_port:

        这是服务器端口号。

_timeout和_enable_inactive_release:

        启动非活跃连接时设置的超时销毁时间 和 销毁标志。

EventLoop:

        这个是主线程对应的事件管理对象。有可能客户端多个客户端对应一个套接字对象,主线程上的套接字对应一个EventLoop对象。这也体现Reactor模型。

Acceptor:

        这个就是TcpServer模块如何让EventLoop对象去监听,主线程上的套接字有没有新的套接字到来。

LoopThreadPool:

        TcpServer模块对应的线程池,该线程池中的每一个线程都绑定这一个EventLoop对象。

std::unordered_map:

        这是用_next_id将Connection对象标识起来,存放到容器中,方便查找。

TcpServer模块整体代码:

class TcpServer {     private:         uint64_t _next_id;      //这是一个自动增长的连接ID,         int _port;         int _timeout;           //这是非活跃连接的统计时间---多长时间无通信就是非活跃连接         bool _enable_inactive_release;//是否启动了非活跃连接超时销毁的判断标志         EventLoop _baseloop;    //这是主线程的EventLoop对象,负责监听事件的处理         Acceptor _acceptor;    //这是监听套接字的管理对象         LoopThreadPool _pool;   //这是从属EventLoop线程池         std::unordered_map _conns;//保存管理所有连接对应的shared_ptr对象          using ConnectedCallback = std::function;         using MessageCallback = std::function;         using ClosedCallback = std::function;         using AnyEventCallback = std::function;         using Functor = std::function;         ConnectedCallback _connected_callback;         MessageCallback _message_callback;         ClosedCallback _closed_callback;         AnyEventCallback _event_callback;     private:         void RunAfterInLoop(const Functor &task, int delay) {             _next_id++;             _baseloop.TimerAdd(_next_id, delay, task);         }         //为新连接构造一个Connection进行管理         void NewConnection(int fd) {             _next_id++;             PtrConnection conn(new Connection(_pool.NextLoop(), _next_id, fd));             conn->SetMessageCallback(_message_callback);             conn->SetClosedCallback(_closed_callback);             conn->SetConnectedCallback(_connected_callback);             conn->SetAnyEventCallback(_event_callback);             conn->SetSrvClosedCallback(std::bind(&TcpServer::RemoveConnection, this, std::placeholders::_1));             if (_enable_inactive_release) conn->EnableInactiveRelease(_timeout);//启动非活跃超时销毁             conn->Established();//就绪初始化             _conns.insert(std::make_pair(_next_id, conn));         }         void RemoveConnectionInLoop(const PtrConnection &conn) {             int id = conn->Id();             auto it = _conns.find(id);             if (it != _conns.end()) {                 _conns.erase(it);             }         }         //从管理Connection的_conns中移除连接信息         void RemoveConnection(const PtrConnection &conn) {             _baseloop.RunInLoop(std::bind(&TcpServer::RemoveConnectionInLoop, this, conn));         }     public:         TcpServer(int port):             _port(port),              _next_id(0),              _enable_inactive_release(false),              _acceptor(&_baseloop, port),             _pool(&_baseloop) {             _acceptor.SetAcceptCallback(std::bind(&TcpServer::NewConnection, this, std::placeholders::_1));             _acceptor.Listen();//将监听套接字挂到baseloop上         }         void SetThreadCount(int count) { return _pool.SetThreadCount(count); }         void SetConnectedCallback(const ConnectedCallback&cb) { _connected_callback = cb; }         void SetMessageCallback(const MessageCallback&cb) { _message_callback = cb; }         void SetClosedCallback(const ClosedCallback&cb) { _closed_callback = cb; }         void SetAnyEventCallback(const AnyEventCallback&cb) { _event_callback = cb; }         void EnableInactiveRelease(int timeout) { _timeout = timeout; _enable_inactive_release = true; }         //用于添加一个定时任务         void RunAfter(const Functor &task, int delay) {             _baseloop.RunInLoop(std::bind(&TcpServer::RunAfterInLoop, this, task, delay));         }         void Start() { _pool.Create();  _baseloop.Start(); } };

相关内容

热门资讯

安卓系统为什么没网络,探究原因... 手机没网络,这可真是让人头疼的小麻烦!你有没有遇到过这种情况:手机屏幕上显示着“无网络连接”,而你明...
安卓苹果换系统安装教程,安卓与... 亲爱的手机控们,是不是觉得手机用久了,系统卡得像蜗牛爬?别急,今天就来教你怎么给安卓和苹果手机换上全...
魅族安卓系统下载软件,尽享智能... 你有没有发现,最近手机圈里又掀起了一股热潮?没错,就是魅族的新款手机!这款手机不仅外观时尚,性能强大...
平板电脑安卓系统12墨,平板电... 亲爱的读者们,你是否也和我一样,对科技新品的到来充满期待?今天,我要和你聊聊一款让人眼前一亮的新品—...
荣耀畅玩刷安卓系统,解锁更多可... 你有没有想过,你的荣耀畅玩手机,其实可以焕发第二春呢?没错,就是刷上全新的安卓系统!想象你的手机瞬间...
为什么安卓系统老是卡机,性能瓶... 手机卡顿真是让人头疼!尤其是安卓系统,有时候用着用着就突然卡住了,让人忍不住想摔手机。那么,为什么安...
哪款盒子是安卓系统,智能娱乐新... 你有没有想过,在这个智能设备横行的时代,哪款盒子是安卓系统最让人心动呢?安卓系统以其开放性和强大的兼...
安卓Q删除系统文件,安全操作与... 亲爱的安卓用户们,你是否曾在使用安卓手机时,不小心误删了重要的系统文件,心里直发慌?别担心,今天就来...
安卓12系统怎么下载谷歌,安卓... 你有没有听说安卓12系统已经发布了?是不是也想赶紧升级体验一下新系统的魅力呢?不过,别急,升级之前你...
安卓系统怎么清除后台,安卓系统... 手机后台程序太多,是不是感觉手机越来越卡?别急,今天就来教你怎么轻松清除安卓系统的后台程序,让你的手...
ios和安卓重置系统,轻松恢复... 手机用久了是不是感觉卡得要命?别急,今天就来给你揭秘如何给iOS和安卓手机来个彻底的重置,让它焕发新...
华为的手机系统与安卓,融合与创... 亲爱的读者们,你是否曾好奇过,为什么华为的手机系统能在众多安卓手机中独树一帜?今天,就让我们一起揭开...
安卓系统如何加通知声音,安卓系... 你有没有发现,手机上的通知声音有时候就像是个小闹钟,总是不请自来地提醒你各种信息。不过,有时候这个“...
安卓系统怎么锁定主屏,安卓系统... 你是不是也和我一样,手机里藏着不少小秘密,不想让别人轻易窥探呢?别急,今天就来教你怎么给安卓手机的主...
安卓系统激活提示谷歌,谷歌助你... 你刚刚入手了一台全新的安卓手机,是不是兴奋得手舞足蹈?不过,别急着高兴,激活手机之前,可别忘了谷歌的...
缤越导航安卓系统,智能出行新体... 你有没有发现,现在汽车导航系统越来越智能了?这不,缤越导航安卓系统就让我眼前一亮。想象坐在车里,手指...
安卓系统好的阅读软件,精选阅读... 你有没有发现,手机里装了那么多应用,阅读软件可是占据了我们日常使用时间的大头呢!安卓系统上的阅读软件...
统信安卓系统下载,下载与体验新... 你有没有想过,手机系统也能像换衣服一样随心所欲地换?没错,今天就要给你揭秘一个超级实用的秘密——统信...
华为新系统鸿蒙安卓,华为安卓融... 你知道吗?最近科技圈可是炸开了锅,华为的新操作系统鸿蒙系统,竟然和安卓系统来了个亲密接触!这可不是一...
安卓系统拍证件照,轻松搞定 你有没有发现,现在拍照已经成为我们生活中不可或缺的一部分?无论是记录生活点滴,还是办理各种证件,拍照...