muduo_net库源码分析(五)(通过EventLoop::runInLoop实现跨线程调用)

2023-11-30 08:38

本文主要是介绍muduo_net库源码分析(五)(通过EventLoop::runInLoop实现跨线程调用),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

知识点

在这里插入图片描述
pipe:等待线程关注fd[0]的可读事件,通知线程只需要往fd[1]中写入数据,fd[0]就变得可读了。
socketpair:与pipe类似,区别在于socketpair可以双向工作,pipe只能单向工作。
eventfd:比pipe更高效。
别忘了线程间的等待/通知还可以用条件变量实现。
在这里插入图片描述
一.Eventloop与Channel是聚合关系,但是有一个地方例外,这个地方就是Eventloop中等待/通知事件的Channel(即wakeupChannel_),与Eventloop是组合关系。
二.EventLoop::wakeup函数中产生了等待/通知读事件,使文件描述符wakeupFd_变得可读:
在这里插入图片描述
三.EventLoop的构造函数:
在这里插入图片描述

四.使用EventLoop::wakeup的原因
唤醒当前IO线程,因为当前线程有可能正阻塞在EventLoop::loop函数中的::poll或::epoll_wait上;
当调用EventLoop::wakeup后,马上产生了等待/通知读事件,这样就会使::poll或::epoll_wait立刻返回

在这里插入图片描述
在这里插入图片描述
对上面两张图的理解要结合EventLoop::loop函数:
在这里插入图片描述
当到来的IO事件处理完毕后,调用EventLoop::doPendingFunctors,这样设计是为了更加灵活,让IO线程也能处理一些计算任务。而EventLoop::runInLoop就是对外的公共接口函数,可跨线程调用,用来向IO线程布置计算任务,具体的任务执行仍然由IO线程完成,因此EventLoop::runInLoop不使用锁就实现了线程安全。
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

源码

首先,结合本节所学内容,对TimerQueue.cc进行如下优化,这样TimerQueue::addTimer和TimerQueue::cancel就支持跨线程调用了,且线程安全,原因如下:
在muduo_net库源码分析(四)中,TimerQueue::addTimer中直接调用了TimerQueue::addTimerInLoop,而TimerQueue::addTimerInLoop不能跨线程调用,只能在当前IO线程中调用(再次强调,每个线程至多一个EventLoop对象,创建了EventLoop对象的线程就是一个IO线程),导致TimerQueue::addTimer也无法跨线程调用;在本节中,由于使用EventLoop::runInLoop给IO线程布置计算任务(任务函数为TimerQueue::addTimerInLoop),而EventLoop::runInLoop是支持跨线程调用的,且线程安全,所以TimerQueue::addTimer就支持跨线程调用了,且线程安全
在这里插入图片描述

EventLoop.h

#ifndef MUDUO_NET_EVENTLOOP_H
#define MUDUO_NET_EVENTLOOP_H#include <vector>#include <boost/noncopyable.hpp>
#include <boost/scoped_ptr.hpp>#include <muduo/base/Mutex.h>
#include <muduo/base/Thread.h>
#include <muduo/base/Timestamp.h>
#include <muduo/net/Callbacks.h>
#include <muduo/net/TimerId.h>namespace muduo
{
namespace net
{class Channel;
class Poller;
class TimerQueue;
///
/// Reactor, at most one per thread.
///
/// This is an interface class, so don't expose too much details.
class EventLoop : boost::noncopyable
{public:typedef boost::function<void()> Functor;EventLoop();~EventLoop();  // force out-line dtor, for scoped_ptr members.////// Loops forever.////// Must be called in the same thread as creation of the object.///void loop();void quit();////// Time when poll returns, usually means data arrivial.///Timestamp pollReturnTime() const { return pollReturnTime_; }/// Runs callback immediately in the loop thread./// It wakes up the loop, and run the cb./// If in the same loop thread, cb is run within the function./// Safe to call from other threads.void runInLoop(const Functor& cb);/// Queues callback in the loop thread./// Runs after finish pooling./// Safe to call from other threads.void queueInLoop(const Functor& cb);// timers////// Runs callback at 'time'./// Safe to call from other threads.///TimerId runAt(const Timestamp& time, const TimerCallback& cb);////// Runs callback after @c delay seconds./// Safe to call from other threads.///TimerId runAfter(double delay, const TimerCallback& cb);////// Runs callback every @c interval seconds./// Safe to call from other threads.///TimerId runEvery(double interval, const TimerCallback& cb);

这篇关于muduo_net库源码分析(五)(通过EventLoop::runInLoop实现跨线程调用)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/436369

相关文章

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

关于集合与数组转换实现方法

《关于集合与数组转换实现方法》:本文主要介绍关于集合与数组转换实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、Arrays.asList()1.1、方法作用1.2、内部实现1.3、修改元素的影响1.4、注意事项2、list.toArray()2.1、方

使用Python实现可恢复式多线程下载器

《使用Python实现可恢复式多线程下载器》在数字时代,大文件下载已成为日常操作,本文将手把手教你用Python打造专业级下载器,实现断点续传,多线程加速,速度限制等功能,感兴趣的小伙伴可以了解下... 目录一、智能续传:从崩溃边缘抢救进度二、多线程加速:榨干网络带宽三、速度控制:做网络的好邻居四、终端交互

java实现docker镜像上传到harbor仓库的方式

《java实现docker镜像上传到harbor仓库的方式》:本文主要介绍java实现docker镜像上传到harbor仓库的方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 前 言2. 编写工具类2.1 引入依赖包2.2 使用当前服务器的docker环境推送镜像2.2

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的

Java easyExcel实现导入多sheet的Excel

《JavaeasyExcel实现导入多sheet的Excel》这篇文章主要为大家详细介绍了如何使用JavaeasyExcel实现导入多sheet的Excel,文中的示例代码讲解详细,感兴趣的小伙伴可... 目录1.官网2.Excel样式3.代码1.官网easyExcel官网2.Excel样式3.代码

Java中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示

python实现对数据公钥加密与私钥解密

《python实现对数据公钥加密与私钥解密》这篇文章主要为大家详细介绍了如何使用python实现对数据公钥加密与私钥解密,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录公钥私钥的生成使用公钥加密使用私钥解密公钥私钥的生成这一部分,使用python生成公钥与私钥,然后保存在两个文

MySQL中的表连接原理分析

《MySQL中的表连接原理分析》:本文主要介绍MySQL中的表连接原理分析,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、背景2、环境3、表连接原理【1】驱动表和被驱动表【2】内连接【3】外连接【4编程】嵌套循环连接【5】join buffer4、总结1、背景

浏览器插件cursor实现自动注册、续杯的详细过程

《浏览器插件cursor实现自动注册、续杯的详细过程》Cursor简易注册助手脚本通过自动化邮箱填写和验证码获取流程,大大简化了Cursor的注册过程,它不仅提高了注册效率,还通过友好的用户界面和详细... 目录前言功能概述使用方法安装脚本使用流程邮箱输入页面验证码页面实战演示技术实现核心功能实现1. 随机