仿Muduo的高并发服务器:LoopThread模块及其ThreadPool模块

本期我们接着深入项目编写

相关代码上传至作者的个人gitee:仿muduo服务器: 本项目致力于实现一个仿造muduo库的简易并发服务器,为个人项目,参考即可喜欢请点个赞谢谢

目录

LoopThread模块

设计思想

源码

LoopThreadPool模块

设计思想

源码


LoopThread模块

设计思想

目标 :将EventLoop模块与线程整合起来

EventLoop模块与线程是一一对应的。EventLoop模块实例化的对象,在构造的时候就会初始化_thread_id,而后边当运行一个操作的时候判断当前是否运行在eventLoop模块对应的线程中,就是将线程ID与EventLoop模块中的thread_id进行一个比较,相同就表示在同一个线程,不同就表示当前运行线程并不是EventLoop线程。

含义 :EventLoop模块在实例化对象的时候,必须在线程内部。EventLoop实例化对象时会设置自己的thread_id。如果我们先创建了多个EventLoop对象,然后创建了多个线程,将各个线程的id,重新给EventLoop进行设置
存在问题 :在构造EventLoop对象,到设置新的thread_id期间将是不可控的。

因此我们必须先创建线程,然后在线程的入口函数中,去实例化EventLoop对象

源码

LoopThread.hpp

cpp 复制代码
#pragma once
#include "EventLoop.hpp"
#include <thread>
#include <mutex>
#include<condition_variable>
#include <atomic>
#include <functional>
namespace ImMuduo
{
    class LoopThread
    {
        public:
            LoopThread();
            ~LoopThread();
            //获取EventLoop智能指针
            EventLoop* GetLoop();
        private:
            //线程入口函数------创建EventLoop对象并启动模块功能
            void ThreadEntry();
            
        private:
            std::thread thread_;//EventLoop线程
            std::unique_ptr<EventLoop> loop_;//EventLoop智能指针,需要在线程内实例化
            //用于实现Loop操作的同步,避免在构造函数中创建EventLoop对象,导致线程创建失败
            std::mutex mutex_;
            std::condition_variable condition_;
    };
}

LoopThread.cpp

cpp 复制代码
#include "LoopThread.hpp"

namespace ImMuduo
{
    LoopThread::LoopThread()
        :thread_(&LoopThread::ThreadEntry, this)
        ,loop_(nullptr)
        ,mutex_()
        ,condition_()
    {}

    LoopThread::~LoopThread()
    {
        if (loop_)
        {
            loop_->Stop();
        }
        if (thread_.joinable())
        {
            thread_.join();
        }
    }

    EventLoop* LoopThread::GetLoop()
    {
        EventLoop* loop = nullptr;
        {
            std::unique_lock<std::mutex> lock(mutex_);
            condition_.wait(lock, [this]() { return loop_ != nullptr; });
            loop = loop_.get();
        }
        return loop;
    }

    void LoopThread::ThreadEntry()
    {
        auto loop = std::make_unique<EventLoop>();
        {
            std::unique_lock<std::mutex> lock(mutex_);
            loop_ = std::move(loop);
            condition_.notify_one();
        }
        loop_->Start();
    }
}

LoopThreadPool模块

设计思想

针对LoopThread设计一个线程池:

LoopThreadPool模块:对所有的LoopThread进行管理及分配
功能

  1. 线程数量可配置(0个或多个)

注意事项:在服务器中,主从Reactor模型是主线程只负责新连接获取,从属线程负责新连接的事件监控及处理

因此当前的线程池,有可能从属线程会数量为0,也就是实现单Reactor服务器,一个线程及负责获取连接,也负责连接的处理

  1. 对所有的线程进行管理,其实就是管理0个或多个LoopThread对象

  2. 提供线程分配的功能

当主线程获取了一个新连接,需要将新连接挂到从属线程上进行事件监控及处理

假设有0个从属线程,则直接分配给主线程的EventLoop,进行处理

假设有多个从属线程,则采用RR轮转思想,进行线程的分配(将对应线程的EventLoop获取到,设置给对应的Connection)

源码

LoopThreadPool.hpp

cpp 复制代码
#pragma once
#include"LoopThread.hpp"
namespace ImMuduo
{
    class LoopThreadPool
    {
        public:
            LoopThreadPool(EventLoop* eventLoop);
            ~LoopThreadPool();
            void SetThreadNum(int thread_num);
            void CreateThreads();
            EventLoop* NextLoop();
        private:
            EventLoop* loop_;
            int next_idx_;
            int thread_num_;
            std::vector<std::unique_ptr<LoopThread>> threads_;
            std::vector<EventLoop*> loops_;
    };
}

LoopThreadPool.cpp

cpp 复制代码
#include"LoopThreadPool.hpp"
namespace ImMuduo
{
    LoopThreadPool::LoopThreadPool(EventLoop* eventLoop)
        :next_idx_(0), thread_num_(0), loop_(eventLoop)
    {}
    LoopThreadPool::~LoopThreadPool()
    {
    }

    void LoopThreadPool::SetThreadNum(int thread_num)
    {
        thread_num_ = thread_num;
    }
    void LoopThreadPool::CreateThreads()
    {
        if (thread_num_ <= 0) return;
        threads_.reserve(thread_num_);
        loops_.reserve(thread_num_);
        for (int i = 0; i < thread_num_; ++i)
        {
            auto t = std::make_unique<LoopThread>();
            loops_.push_back(t->GetLoop());
            threads_.push_back(std::move(t));
        }
    }

    EventLoop* LoopThreadPool::NextLoop()
    {
        if (loops_.empty()) return loop_;
        EventLoop* ret = loops_[next_idx_];
        next_idx_ = (next_idx_ + 1) % static_cast<int>(loops_.size());
        return ret;
    }
}

本期内容到这里就结束了,喜欢请点个赞谢谢

封面图自取:

相关推荐
二月龙2 小时前
微信小程序页面栈限制解析与突破方案
后端
Rust研习社2 小时前
你为什么总是入门 Rust 失败
开发语言·后端·rust
SamDeepThinking2 小时前
批评下属不如当场展示解决方案
后端·程序员·团队管理
techdashen3 小时前
等了两年,Cloudflare 终于给规则引擎加上了通配符
服务器·rust
zhangfeng11333 小时前
宝塔服务器完全可以安装 Git,进行版本管理,而且非常简单
运维·服务器·人工智能·git·编程
盐焗鹌鹑蛋3 小时前
【C++】模板进阶
c++
菜鸟的日志3 小时前
【嵌入系统】嵌入式学习笔记(一)
windows·笔记·嵌入式硬件·学习·ubuntu·操作系统
繁星蓝雨3 小时前
Qt多界面创建的优化问题(main函数或主界面中创建?)—————附带详细方法
c++·qt·架构·多界面管理
AskHarries3 小时前
GPT-Image-2(img2)到底能做什么?
后端