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

相关文章

C++使用栈实现括号匹配的代码详解

《C++使用栈实现括号匹配的代码详解》在编程中,括号匹配是一个常见问题,尤其是在处理数学表达式、编译器解析等任务时,栈是一种非常适合处理此类问题的数据结构,能够精确地管理括号的匹配问题,本文将通过C+... 目录引言问题描述代码讲解代码解析栈的状态表示测试总结引言在编程中,括号匹配是一个常见问题,尤其是在

Python调用Orator ORM进行数据库操作

《Python调用OratorORM进行数据库操作》OratorORM是一个功能丰富且灵活的PythonORM库,旨在简化数据库操作,它支持多种数据库并提供了简洁且直观的API,下面我们就... 目录Orator ORM 主要特点安装使用示例总结Orator ORM 是一个功能丰富且灵活的 python O

Java实现检查多个时间段是否有重合

《Java实现检查多个时间段是否有重合》这篇文章主要为大家详细介绍了如何使用Java实现检查多个时间段是否有重合,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录流程概述步骤详解China编程步骤1:定义时间段类步骤2:添加时间段步骤3:检查时间段是否有重合步骤4:输出结果示例代码结语作

使用C++实现链表元素的反转

《使用C++实现链表元素的反转》反转链表是链表操作中一个经典的问题,也是面试中常见的考题,本文将从思路到实现一步步地讲解如何实现链表的反转,帮助初学者理解这一操作,我们将使用C++代码演示具体实现,同... 目录问题定义思路分析代码实现带头节点的链表代码讲解其他实现方式时间和空间复杂度分析总结问题定义给定

Java覆盖第三方jar包中的某一个类的实现方法

《Java覆盖第三方jar包中的某一个类的实现方法》在我们日常的开发中,经常需要使用第三方的jar包,有时候我们会发现第三方的jar包中的某一个类有问题,或者我们需要定制化修改其中的逻辑,那么应该如何... 目录一、需求描述二、示例描述三、操作步骤四、验证结果五、实现原理一、需求描述需求描述如下:需要在

如何使用Java实现请求deepseek

《如何使用Java实现请求deepseek》这篇文章主要为大家详细介绍了如何使用Java实现请求deepseek功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1.deepseek的api创建2.Java实现请求deepseek2.1 pom文件2.2 json转化文件2.2

Java调用DeepSeek API的最佳实践及详细代码示例

《Java调用DeepSeekAPI的最佳实践及详细代码示例》:本文主要介绍如何使用Java调用DeepSeekAPI,包括获取API密钥、添加HTTP客户端依赖、创建HTTP请求、处理响应、... 目录1. 获取API密钥2. 添加HTTP客户端依赖3. 创建HTTP请求4. 处理响应5. 错误处理6.

python使用fastapi实现多语言国际化的操作指南

《python使用fastapi实现多语言国际化的操作指南》本文介绍了使用Python和FastAPI实现多语言国际化的操作指南,包括多语言架构技术栈、翻译管理、前端本地化、语言切换机制以及常见陷阱和... 目录多语言国际化实现指南项目多语言架构技术栈目录结构翻译工作流1. 翻译数据存储2. 翻译生成脚本

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

如何通过Python实现一个消息队列

《如何通过Python实现一个消息队列》这篇文章主要为大家详细介绍了如何通过Python实现一个简单的消息队列,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录如何通过 python 实现消息队列如何把 http 请求放在队列中执行1. 使用 queue.Queue 和 reque