基于C++实现的EventLoop与事件驱动编程

一,概念介绍

事件驱动编程(Event-Driven)是一种编码范式,常被应用在图形用户界面,应用程序,服务器开发等场景。

采用事件驱动编程的代码中,通常要有事件循环,侦听事件,以及不同事件所对应的回调函数。

事件驱动编程经常被应用在前端开发以及C++服务器开发等场景。

Event即事件,是事件驱动编程中的基本处理单元,可以理解为各种各样的信号,对于UI界面来说,鼠标点击、键盘输入、触摸屏输入都可以理解为事件。

事件循环模式(Event loop)是一种简单且高效的并发编程模式,当前业界有很多主流的C++编程框架比如libevent,libuv,Boost.Asio等都支持事件循环机制。但是考虑代码封装上的简洁,我们也可以借助C++11标准实现自己的事件循环代码。通过事件循环,程序可以支持非阻塞的异步操作,提高系统的性能。

步骤示意图:

拿Event填充Event队列:

客户端只管发起请求,触发相应的事件,其他步骤交给队列去处理:

EventLoop样例代码:

#include <algorithm>
#include <iostream>
#include <vector>
#include <map>class EventManager {
private:std::map<std::string, std::vector<void (*)(int)> > events;public:EventManager() {}EventManager* eventRegist(std::string event_name, void (*callback)(int)) {std::vector<void (*)(int)>* listeners = &events[event_name];// if this listener is already registered, we wont add it againif (std::find(listeners->begin(), listeners->end(), callback) !=  listeners->end()) {return this;}listeners->push_back(callback);return this;}bool emit(std::string event_name, int arg) {std::vector<void (*)(int)> listeners = events[event_name];if (listeners.size() == 0) return false;for (int idx = 0; idx < listeners.size(); idx += 1) {listeners[idx](arg);}return true;}
};void callback1(int num) {std::cout << "callback1-" << num << std::endl;
}
void callback2(int num) {std::cout << "callback2-" << num << std::endl;
}int main() {EventManager* event_manager = new EventManager();//注册回调函数event_manager->eventRegist("event1", callback1);event_manager->eventRegist("event2", callback2);//执行回调函数int eventA = event_manager->emit("event1", 10);int eventB = event_manager->emit("event2", 20);return 0;
}

运行结果:

callback1-10
callback2-20

根据以上代码样例,我们发现事件驱动编程通常有以下编码元素:

1.回调函数:回调函数可以是预定义的函数,也可以是匿名函数或Lambda表达式。

2.注册回调:将回调函数赋值给Event的一个std::function成员变量,再将Event添加到Event Loop对应的队列中。

3.触发Event对应的请求以后,从队列中执行事件对应的回调函数。

二,Event Loop步骤拆解

事件循环(Event loop)是一种轮询机制,这种轮询是异步的,有时候轮询和事件注册发生在不同的线程中。

事件循环特别适用于异步编程,在事件循环中,程序会不断地等待事件的发生,并根据事件的类型和优先级来执行相应的处理逻辑。

事件循环主要由以下四个部分组成:

1.事件队列(Event Queue):

用于存储待处理的事件,每个事件都包含一个回调函数和相应的函数参数。

2.事件触发器(Event Trigger):

负责监听外部事件(如用户输入、网络请求等),并将事件添加到事件队列中。

3.事件处理器(Event Handler):

从事件队列中取出对应事件,并执行事件的回调函数。

4.回调函数(Callback Function):

与特定事件相关联的函数,当对应的事件发生时才会被调用执行。回调函数只有被"注册"到事件队列中才会被调用执行。所谓的"注册"就是将回调函数赋值给Event对应的函数对象。

事件循环(Event Loop)是一个无限循环,它会不断地从事件队列中取出事件,并执行对应的回调函数。在有些模式下,事件循环会检查事件队列是否为空,如果为空则会进入休眠状态等待新的事件到来。

c++服务器开发教程

【腾讯T9推荐】2024最新linux c/c++后端服务器开发教程,通俗易懂深入底层讲解,多项目实战,1V1指导,学完轻松拿下大厂offer!!!icon-default.png?t=N7T8https://www.bilibili.com/video/BV1XZ421q7UD/

免费学习地址:c/c++ linux服务器开发/后台架构师

需要C/C++ Linux服务器架构师学习资料加qun579733396获取(资料包括C/C++,Linux,golang技术,Nginx,ZeroMQ,MySQL,Redis,fastdfs,MongoDB,ZK,流媒体,CDN,P2P,K8S,Docker,TCP/IP,协程,DPDK,ffmpeg等),免费分享

事件循环的基本流程如下:

step.01:初始化事件队列。

step.02:进入循环,等待事件的发生。

step.03:当监听的事件被触发时,将事件添加到事件队列中。

step.04:从事件队列中取出一个事件,并异步执行对应的回调函数。

step.05:返回第2步,继续等待下一个事件的发生。

注意:step.01~step.05并不只发生在同一个线程中,很多时候,回调函数的调用会放在子线程中进行。

参考以上步骤,我们可以设计以下Event Loop代码:

#include <condition_variable>
#include <functional>
#include <thread>
#include <vector>
#include <iostream>class EventLoop
{
public:using callable_t = std::function<void()>;EventLoop() = default;~EventLoop() noexcept{enqueue([this]{m_running = false;});std::cout << "step.02: other thread print.\n";m_thread.join();}//禁用移动构造 & 拷贝构造EventLoop(const EventLoop&) = delete;EventLoop(const EventLoop&&) = delete;EventLoop& operator= (const EventLoop&) = delete;EventLoop& operator= (const EventLoop&&)  = delete;void enqueue(callable_t&& callable) noexcept{{std::lock_guard<std::mutex> guard(m_mutex);m_writeBuffer.emplace_back(std::move(callable));}m_condVar.notify_one();}
private:std::vector<callable_t> m_writeBuffer;std::mutex m_mutex;std::condition_variable m_condVar;bool m_running{ true };std::thread m_thread{ &EventLoop::threadFunc, this};void threadFunc() noexcept{std::vector<callable_t> readBuffer;while (m_running){{std::unique_lock<std::mutex> lock(m_mutex);m_condVar.wait(lock, [this]{return !m_writeBuffer.empty();});std::swap(readBuffer, m_writeBuffer);}for (callable_t& func : readBuffer){func();}readBuffer.clear();}std::cout << "step.03: event loop end.\n";}
};int main()
{EventLoop eventLoop;eventLoop.enqueue([]{std::cout << "Event_01 is running.\n";});eventLoop.enqueue([]{std::cout << "Event_02 is running.\n";});eventLoop.enqueue([]{std::cout << "Event_03 is running.\n";});std::cout << "step.01: main thread print.\n";
}

运行结果:

step.01: main thread print.
step.02: other thread print.
Event_01 is running.
Event_02 is running.
Event_03 is running.
step.03: event loop end.

三,事件驱动代码实战

Demo1:没有添加Event Loop,主要运行Callback回调函数

#include <iostream>
#include <functional>
#include <string>
// 定义回调函数类型
typedef std::function<void(std::string str)> Callback;
// 模拟用户输入事件
void simulateUserInput(Callback callback_func) {std::string input;std::cout << "请输入一段文字:";getline(std::cin, input);callback_func(input);  // 触发回调函数
}
// 处理用户输入事件的回调函数
void handleUserInput(std::string inputStr) {std::cout << "用户输入事件已触发!" << std::endl;std::cout << "用户输入的是: " << inputStr << std::endl;return;
}int main() {simulateUserInput(handleUserInput);return 0;
}

运行结果:

请输入一段文字:hello
用户输入事件已触发!
用户输入的是:hello

Demo2:

#include <iostream>
#include <functional>
#include <queue>
//定义事件类型
typedef std::function<void()> Event;
//事件队列
std::queue<Event> eventQueue;
//注册回调函数
void registerEventHandler(Event event) {eventQueue.push(event);
}
//事件处理器
void processEvents() {while (!eventQueue.empty()) {Event event = eventQueue.front();event();  //调用回调函数eventQueue.pop();}
}
//回调函数
void callback1() {std::cout << "Callback 1 called" << std::endl;
}
void callback2() {std::cout << "Callback 2 called" << std::endl;
}
int main() {//注册回调函数到事件队列registerEventHandler(callback1);registerEventHandler(callback2);//处理事件processEvents();return 0;
}

运行结果:

Callback 1 called
Callback 2 called

Demo3:

#include <iostream>
#include <functional>
#include <queue>
//定义Event结构体
struct Event {std::function<void()> callback;
};
//定义事件处理器
class EventHandler {
public:void handleEvent(Event event) {event.callback();}
};
//定义事件循环
class EventLoop {
public:void addEvent(Event event) {eventQueue.push(event);}void run() {while (!eventQueue.empty()) {Event event = eventQueue.front();eventQueue.pop();eventHandler.handleEvent(event);}}
private:std::queue<Event> eventQueue;EventHandler eventHandler;
};
int main() {//创建事件循环对象EventLoop eventLoop;//回调函数std::function<void()> callback1 = []() {std::cout << "Event 1 triggered!" << std::endl;};std::function<void()> callback2 = []() {std::cout << "Event 2 triggered!" << std::endl;};//创建事件并添加到事件循环中Event event1{ callback1 };Event event2{ callback2 };eventLoop.addEvent(event1);eventLoop.addEvent(event2);//运行事件循环eventLoop.run();return 0;
}

运行结果:

Event 1 triggered!
Event 2 triggered!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/368152.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

无线物联网题集

测试一 未来信息产业的发展在由信息网络向 全面感知和 智能应用两个方向拓展、延伸和突破。 各国均把 物联网作为未来信息化战略的重要内容,融合各种信息技术,突破互联网的限制,将物体接入信息网络。 计算机的出现,开始了第四次工业革命,开始了人机物的高度融合&#xff08;&…

数据库安全审计系统:满足数据安全治理合规要求

伴随着数据库信息价值以及可访问性提升&#xff0c;使得数据库面对来自内部和外部的安全风险大大增加&#xff0c;如违规越权操作、恶意入侵导致机密信息窃取泄漏&#xff0c;但事后却无法有效追溯和审计。 国内专注于保密与非密领域的分级保护、等级保护、业务连续性安全和大数…

微信小程序 typescript 开发日历界面

1.界面代码 <view class"o-calendar"><view class"o-calendar-container" ><view class"o-calendar-titlebar"><view class"o-left_arrow" bind:tap"prevMonth">《</view>{{year}}年{{month…

Kafka集群安装部署

简介 Kafka是一款分布式的、去中心化的、高吞吐低延迟、订阅模式的消息队列系统。 同RabbitMQ一样&#xff0c;Kafka也是消息队列。不过RabbitMQ多用于后端系统&#xff0c;因其更加专注于消息的延迟和容错。 Kafka多用于大数据体系&#xff0c;因其更加专注于数据的吞吐能力…

机器学习:预测评估8类指标

机器学习&#xff1a;8类预测评估指标 R方值、平均值绝对误差值MAE、均方误差MSE、均方误差根EMSE、中位数绝对误差MAD、平均绝对百分误差MAPE、可解释方差分EVS、均方根对数误差MLSE。 一、R方值 1、说明&#xff1a; R方值&#xff0c;也称为确定系数或拟合优度&#xff…

css样式flex布局之,盒子垂直居中

<div class"item"><img src"../../assets/images!code_app.png" alt"" /><div>5555</div><p>微信扫一扫关注</p><p>“快速预约挂号”</p></div>.item{display: flex;flex-direction: col…

高薪程序员必修课-Java中 ReentrantLock的公平锁和非公平锁底层实现原理

目录 前言 公平锁&#xff08;Fair Lock&#xff09; 原理 实现 示例代码 底层实现 非公平锁&#xff08;Non-Fair Lock&#xff09; 原理 实现 示例代码 底层实现 比较与选择 总结 ⭐️ 好书推荐 前言 在Java中&#xff0c;ReentrantLock 提供了公平锁和非公平锁…

taoCMS v3.0.2 文件上传漏洞(CVE-2022-23880)

前言 CVE-2022-23880是一个影响taoCMS v3.0.2的任意文件上传漏洞。攻击者可以利用此漏洞通过上传特制的PHP文件在受影响的系统上执行任意代码。 漏洞细节 描述: 在taoCMS v3.0.2的文件管理模块中存在任意文件上传漏洞。攻击者可以通过上传恶意的PHP文件来执行任意代码。 影响…

光伏电站数据采集方案(基于工业路由器部署)

​ 一、方案概述 本方案采用星创易联SR500工业路由器作为核心网关设备&#xff0c;实现对光伏电站现场数据的实时采集、安全传输和远程监控。SR500具备多接口、多功能、高可靠性等特点&#xff0c;能够满足光伏电站数据采集的各种需求。&#xff08;key-iot.com/iotlist/sr500…

如何策划交互设计创意?( 计育韬老师高校公益巡讲答疑实录2024)

这是计育韬老师第 8 次开展面向全国高校的新媒体技术公益巡讲活动了。而在每场讲座尾声&#xff0c;互动答疑环节往往反映了高校师生当前最普遍的运营困境&#xff0c;特此计老师在现场即兴答疑之外&#xff0c;会尽量选择有较高价值的提问进行文字答疑梳理。 *本轮巡讲主题除了…

【Altium】AD-在原理图中如何绘制贝塞尔曲线

【更多软件使用问题请点击亿道电子官方网站】 1、 文档目标 在原理图中绘制贝塞尔曲线的方法 2、 问题场景 贝塞尔曲线主要用来描述各种波形曲线&#xff0c;如正弦、余弦曲线等。贝塞尔曲线的绘制和直线类似&#xff0c;需要固定多个顶点&#xff08;最少4个&#xff09;后即…

详解 RisePro 信息窃密木马

RisePro 是一种窃密木马&#xff0c;以恶意软件即服务&#xff08;MaaS&#xff09;的模式在地下论坛出售。该恶意软件家族最早在 2022 年被发现&#xff0c;近期攻击行为快速增长。 RisePro 不依赖特定的感染媒介&#xff0c;可以通过多种方式植入失陷主机&#xff0c;通常使…

Zabbix 配置WEB监控

Zabbix WEB监控介绍 在Zabbix中配置Web监控&#xff0c;可以监控网站的可用性和响应时间。Zabbix提供了内置的Web监控功能&#xff0c;通过配置Web场景&#xff08;Web Scenario&#xff09;&#xff0c;可以监控HTTP/HTTPS协议下的Web服务。 通过Zabbix的WEB监控可以监控网站…

Java的NIO体系

目录 NIO1、操作系统级别下的IO模型有哪些&#xff1f;2、Java语言下的IO模型有哪些&#xff1f;3、Java的NIO应用场景&#xff1f;相比于IO的优势在哪&#xff1f;4、Java的IO、NIO、AIO 操作文件读写5、NIO的核心类 :Buffer&#xff08;缓冲区&#xff09;、Channel&#xff…

【代码随想录——图论——图论理论基础】

1. 图论理论基础 1.1 图的基本概念 二维坐标中&#xff0c;两点可以连成线&#xff0c;多个点连成的线就构成了图。 当然图也可以就一个节点&#xff0c;甚至没有节点&#xff08;空图&#xff09; 1.1.1 图的种类 有向图 加权有向图无权有向图 无向图 加权无向图无权无向…

.NET C# 使用OpenCV实现人脸识别

.NET C# 使用OpenCV实现模型训练、人脸识别 码图~~~ 1 引入依赖 OpenCvSHarp4 - 4.10.0.20240616 OpenCvSHarp4.runtime.win - 4.10.0.20240616 2 人脸数据存储结构 runtime directory | face | {id}_{name} | *.jpg id - 不可重复 name - 人名 *.jpg - 人脸照片3 Demo 3.…

松下Panasonic机器人维修故障原因

松下机器人伺服电机是许多工业自动化设备的关键组成部分。了解如何进行Panasonic工业机械臂电机维修&#xff0c;对于确保设备正常运行至关重要。 【松下焊接机器人维修案例】【松下机器人维修故障排查】 一、常见松下工业机械手伺服电机故障及原因 1. 过热&#xff1a;过热可…

vue2+element-ui新增编辑表格+删除行

实现效果&#xff1a; 代码实现 &#xff1a; <el-table :data"dataForm.updateData"border:header-cell-style"{text-align:center}":cell-style"{text-align:center}"><el-table-column label"选项字段"align"center&…

C++:求梯形面积

梯形面积 已知上底15厘米&#xff0c;下底25厘米&#xff0c;问梯形面积值是多少&#xff1f; #include<iostream> using namespace std; int main() {//梯形的面积公式&#xff08;上底下底&#xff09; 高 2//上底变量、下底变量int s,d,h,m;s15;d25;h 2*150 * 2/s ;…

相亲交友APP系统婚恋交友社交软件开发语音视频聊天平台定制开发-婚恋相亲交友软件平台介绍——app小程序开发定制

互联网飞速发展的时代&#xff0c;相亲交友软件成为了许多年轻人首选的相亲方式&#xff0c;越来越多的单身男女希望在婚恋交友软件平台上寻找灵魂伴侣&#xff0c;相亲交友软件因此具有很高的市场价值。 多客婚恋相亲交友系统是一款定位高端&#xff0c;到手就能运营的成熟婚恋…