事件驱动和消息驱动的对比_事件驱动与消息驱动的区别-程序员宅基地

技术标签: C++  

通常,我们写服务器处理模型的程序时,有以下几种模型:
(1)每收到一个请求,创建一个新的进程,来处理该请求;
(2)每收到一个请求,创建一个新的线程,来处理该请求;
(3)每收到一个请求,放入一个事件列表,让主进程通过非阻塞I/O方式来处理请求
上面的几种方式,各有千秋,
第(1)中方法,由于创建新的进程的开销比较大,所以,会导致服务器性能比较差,但实现比较简单。
第(2)种方式,由于要涉及到线程的同步,有可能会面临死锁等问题。
第(3)种方式,在写应用程序代码时,逻辑比前面两种都复杂。
综合考虑各方面因素,一般普遍认为第(3)种方式是大多数网络服务器采用的方式
 
看图说话讲事件驱动模型
在UI编程中,常常要对鼠标点击进行相应,首先如何获得鼠标点击呢?
方式一:创建一个线程,该线程一直循环检测是否有鼠标点击,那么这个方式有以下几个缺点:
1. CPU资源浪费,可能鼠标点击的频率非常小,但是扫描线程还是会一直循环检测,这会造成很多的CPU资源浪费;如果扫描鼠标点击的接口是阻塞的呢?
2. 如果是堵塞的,又会出现下面这样的问题,如果我们不但要扫描鼠标点击,还要扫描键盘是否按下,由于扫描鼠标时被堵塞了,那么可能永远不会去扫描键盘;
3. 如果一个循环需要扫描的设备非常多,这又会引来响应时间的问题;
所以,该方式是非常不好的。

方式二:就是事件驱动模型
目前大部分的UI编程都是事件驱动模型,如很多UI平台都会提供onClick()事件,这个事件就代表鼠标按下事件。事件驱动模型大体思路如下:
1. 有一个事件(消息)队列;
2. 鼠标按下时,往这个队列中增加一个点击事件(消息);
3. 有个循环,不断从队列取出事件,根据不同的事件,调用不同的函数,如onClick()、onKeyDown()等;
4. 事件(消息)一般都各自保存各自的处理函数指针,这样,每个消息都有独立的处理函数;

事件驱动编程是一种编程范式,这里程序的执行流由外部事件来决定。它的特点是包含一个事件循环,当外部事件发生时使用回调机制来触发相应的处理。另外两种常见的编程范式是(单线程)同步以及多线程编程。

让我们用例子来比较和对比一下单线程、多线程以及事件驱动编程模型。下图展示了随着时间的推移,这三种模式下程序所做的工作。这个程序有3个任务需要完成,每个任务都在等待I/O操作时阻塞自身。阻塞在I/O操作上所花费的时间已经用灰色框标示出来了。

在单线程同步模型中,任务按照顺序执行。如果某个任务因为I/O而阻塞,其他所有的任务都必须等待,直到它完成之后它们才能依次执行。这种明确的执行顺序和串行化处理的行为是很容易推断得出的。如果任务之间并没有互相依赖的关系,但仍然需要互相等待的话这就使得程序不必要的降低了运行速度。

在多线程版本中,这3个任务分别在独立的线程中执行。这些线程由操作系统来管理,在多处理器系统上可以并行处理,或者在单处理器系统上交错执行。这使得当某个线程阻塞在某个资源的同时其他线程得以继续执行。与完成类似功能的同步程序相比,这种方式更有效率,但程序员必须写代码来保护共享资源,防止其被多个线程同时访问。多线程程序更加难以推断,因为这类程序不得不通过线程同步机制如锁、可重入函数、线程局部存储或者其他机制来处理线程安全问题,如果实现不当就会导致出现微妙且令人痛不欲生的bug。

在事件驱动版本的程序中,3个任务交错执行,但仍然在一个单独的线程控制中。当处理I/O或者其他昂贵的操作时,注册一个回调到事件循环中,然后当I/O操作完成时继续执行。回调描述了该如何处理某个事件。事件循环轮询所有的事件,当事件到来时将它们分配给等待处理事件的回调函数。这种方式让程序尽可能的得以执行而不需要用到额外的线程。事件驱动型程序比多线程程序更容易推断出行为,因为程序员不需要关心线程安全问题。

当我们面对如下的环境时,事件驱动模型通常是一个好的选择:

程序中有许多任务,而且…
任务之间高度独立(因此它们不需要互相通信,或者等待彼此)而且…
在等待事件到来时,某些任务会阻塞。
当应用程序需要在任务间共享可变的数据时,这也是一个不错的选择,因为这里不需要采用同步处理。

网络应用程序通常都有上述这些特点,这使得它们能够很好的契合事件驱动编程模型。

事件驱动机制跟消息驱动机制相比

事件:按下鼠标,按下键盘,按下游戏手柄,将U盘插入USB接口,都将产生事件。比如说按下鼠标左键,将产生鼠标左键被按下的事件。

 消息:当鼠标被按下,产生了鼠标按下事件,windows侦测到这一事件的发生,随即发出鼠标被按下的消息到消息队列中,这消息附带了一系列相关的事件信息,比如鼠标哪个键被按了,在哪个窗口被按的,按下点的坐标是多少?如此等等。

 

1.要理解事件驱动和程序,就需要与非事件驱动的程序进行比较。实际上,现代的程序大多是事件驱动的,比如多线程的程序,肯定是事件驱动的。早期则存在许多非事件驱动的程序,这样的程序,在需要等待某个条件触发时,会不断地检查这个条件,直到条件满足,这是很浪费cpu时间的。而事件驱动的程序,则有机会释放cpu从而进入睡眠态(注意是有机会,当然程序也可自行决定不释放cpu),当事件触发时被操作系统唤醒,这样就能更加有效地使用cpu.
2.再说什么是事件驱动的程序。一个典型的事件驱动的程序,就是一个死循环,并以一个线程的形式存在,这个死循环包括两个部分,第一个部分是按照一定的条件接收并选择一个要处理的事件,第二个部分就是事件的处理过程。程序的执行过程就是选择事件和处理事件,而当没有任何事件触发时,程序会因查询事件队列失败而进入睡眠状态,从而释放cpu。
3.事件驱动的程序,必定会直接或者间接拥有一个事件队列,用于存储未能及时处理的事件。
4.事件驱动的程序的行为,完全受外部输入的事件控制,所以,事件驱动的系统中,存在大量这种程序,并以事件作为主要的通信方式。
5.事件驱动的程序,还有一个最大的好处,就是可以按照一定的顺序处理队列中的事件,而这个顺序则是由事件的触发顺序决定的,这一特性往往被用于保证某些过程的原子化。
6.目前windows,linux,nucleus,vxworks都是事件驱动的,只有一些单片机可能是非事件驱动的。


事件模式耦合高,同模块内好用;消息模式耦合低,跨模块好用。事件模式集成其它语言比较繁琐,消息模式集成其他语言比较轻松。事件是侵入式设计,霸占你的主循环;消息是非侵入式设计,将主循环该怎样设计的自由留给用户。如果你在设计一个东西举棋不定,那么你可以参考win32的GetMessage,本身就是一个藕合度极低的接口,又足够自由,接口任何语言都很方便,具体应用场景再在其基础上封装成事件并不是难事,接口耦合较低,即便哪天事件框架调整,修改外层即可,不会伤经动骨。而如果直接实现成事件,那就完全反过来了。

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/laobai1015/article/details/107732064

智能推荐

计算机office四级考证试题,一级计算机考试上机WORD试题(三)-程序员宅基地

文章浏览阅读197次。第3题、请在“考试项目”菜单上选择“字处理软件使用”,完成下面的内容:注意:下面出现的“考生文件夹”均为%USER%****** 本套题共有2小题 ******一、在考生文件夹下打开文档WDT31.DOC。操作完成后以原文件名保存文档。(1)将标题段(“分析:超越Linux、Windows之争”)的所有文字设置为三号、黄色、加粗,居中并添加文字蓝色底纹,其中的英文文字设置为Arial Black字..._分析:超越linux计算机一级

C++编程 (三)--- 深入C++后台开发_c++后端开发-程序员宅基地

文章浏览阅读1.1w次,点赞15次,收藏103次。搞了很久搜索了,可是做的很多都是业务逻辑和PM的需求,也没有高大上的技术,我也认真总结和实践了一些深入的技术。总的来说C++后台开发深入一些的有网络编程、多线程编程、进程/线程同步/通信和调度、动态链接库使用、常用的框架的深入阅读和理解、常用的运行时程序问题排查(内存泄露、无法响应新的请求)、分布式系统的使用、高并发系统优化。所以本文一共分为如下九个部分:一、网络编程二、多线程编程三、_c++后端开发

R语言 boxplo函数用法及箱线图介绍_r gg_boxplot_col()-程序员宅基地

文章浏览阅读1.3w次,点赞7次,收藏26次。箱线图(boxplot)介绍箱线图(Boxplot)也称箱须图(Box-whisker Plot),是利用数据中的五个统计量:最小值、第一四分位数、中位数、第三四分位数与最大值来描述数据的一种方法。它也可以粗略地看出数据是否具有有对称性,分布的离散程度等信息;特别适用于对几个样本的比较。注:四分位数(Quartile),即统计学中,把所有数值由小到大排列并分成四等份,处于三个分割点位置的数值就是四分位数。 第一四分位数 (Q1),又称“较小四分位数”,等于该样本中..._r gg_boxplot_col()

Windows安装faceswap_swapface-程序员宅基地

文章浏览阅读1w次。一、安装python3.7版本的anaconda3下载地址 :Anaconda3-2020.02-Windows-x86_64.exe然后安装,一路下一步安装,下边两个勾选上二、在.condarc文件中写入如下内容,配置Anaconda的国内镜像地址channels: - defaultsshow_channel_urls: truedefault_channels: - https://mirrors.tuna.tsinghua.edu.cn/anaconda/pkg.._swapface

com.mysql.cj.jdbc.driver与com.mysql.jdbc.driver'_hive链接mysql怎么把com.mysql.cj.jdbc.driver换成driver-程序员宅基地

文章浏览阅读1.4k次。com.mysql.cj.jdbc.driver'的配置连接如下:jdbc.driver=com.mysql.cj.jdbc.Driverjdbc.url=jdbc:mysql://192.168.1.99:3306/root?useUnicode=true&characterEncoding=UTF-8&serverTimezone=UTC&useSSL=fals..._hive链接mysql怎么把com.mysql.cj.jdbc.driver换成driver

操作系统 第二章【记录型信号量机制、独木桥问题】【MOOC答案】_简述生产者消费者问题的解决方案,并说明用记录型信号量机制解决生产者消费者-程序员宅基地

文章浏览阅读5.2k次,点赞3次,收藏23次。操作系统 第二章【记录型信号量机制、独木桥问题】【MOOC答案】_简述生产者消费者问题的解决方案,并说明用记录型信号量机制解决生产者消费者

随便推点

svn 服务器用户权限设置,mac下配置svn服务器详解及用户的权限管理-程序员宅基地

文章浏览阅读1.1k次。首先,感谢jsntghf和星辰的天空的好文分享,不是他们的文章,我估计须要花费更多的精力和时间。在这里我只是对他们文章的润色和本身测试遇到问题的标注。htmlMac自带了svn服务器和客户端,因此只须要简单配置一下就可使用apache1. 建立svn repository服务器Shell代码微信svnadmincreate/Users/mac22/svn/repositorysvnadmin:..._svn e00002:can't create directory

HMC833 PLL时钟控制芯片读写操作寄存器_directory bread-程序员宅基地

文章浏览阅读1.8w次,点赞5次,收藏35次。然后对芯片进行写数据测试,主要测试数据有两组740M和2270M,740M的测试结果为输出幅度30dBm,且杂散较大,分别为1、2、3倍频,与740M幅度相差为-16dBc,但由于其频率相差加大,可以在后端接滤波器以做改善。原因:在740和2270MHz输出频率的测试之后,还对从 740~2000M依次20M频率步进做了测试,发现杂散的出现主要为当前输出频率的倍频,随着频率的升高,杂散间的频率间隔等比增大,在频率上到1800M时在频谱仪中以3GHZ的Span,观测时已无法观测。(2)SCLK=0;_directory bread

android apk安装工具,安卓装机必备工具!一键批量安装应用apk-程序员宅基地

文章浏览阅读5.3k次。原标题:安卓装机必备工具!一键批量安装应用apk安卓如何批量安装App的apk安装包?这是很多朋友都遇到的问题。安卓可以通过apk安装电子市场所没有的App,不过有时候下载一堆apk安装包回来,还需要一个个点开安装,非常麻烦。有没有什么比较好的方法可以一下子把apk都安装完?当然有!今天,笔者就来为大家介绍一款可以一键批量安装apk的工具,一起来看看吧。软件名称:一键安装APK:1 Click I..._手机apk强制安装器

自定义带下划线的UITextField_uitextfield 下面加横线-程序员宅基地

文章浏览阅读340次。.h#import <UIKit/UIKit.h>NS_ASSUME_NONNULL_BEGIN@interface SelfTextField : UITextField@endNS_ASSUME_NONNULL_END.m#import "SelfTextField.h"#import "CommonUI.h"@interface SelfTextField ()@end@implementat..._uitextfield 下面加横线

送40本高质量AI、Python书籍-程序员宅基地

文章浏览阅读185次。有名人说过:”说不清是哪本书影响了我的追求和命运,唯有感恩,感恩于那个时候读到的每一本书。”在此给大家送一波福利,这次联系了 9个好友一起给各位送书,每个号送 4本,一共..._python ai图书

Java毕业设计-基于ssm的共享型汽车租赁管理系统-第64期-程序员宅基地

文章浏览阅读652次,点赞26次,收藏8次。基于ssm的共享型汽车租赁管理系统:有配套报告文档,前端jsp、jquery、bootstrap,后端 springmvc、spring、mybatis,集成汽车租赁、客户管理、汽车管理、网站公告等功能于一体的系统。

推荐文章

热门文章

相关标签