CEvent类的一个对象,表示一个“事件”——一个允许一个事件发生时线程通知另一个线程的同步对象。在一个线程需要了解何时执行任务时,事件是十分有用的。例如,拷贝数据到数据文档时,线程应被通知何时数据是可用的。当新数据可用时,通过运用CEvent对象来通知拷贝线程,线程才可能尽快地执行。例如在某些网络应用程序中,一个线程(记为A)负责监听通信端口,另一个线程(记为B)负责更新用户数据。通过使用CEvent类,线程A可以通知线程B何时更新用户数据,这样线程B可以尽快地更新用户数据。

CEvent对象有两种类型:自动和手工。一个手工CEvent对象存在于由ResetEvent或SetEvent设置的状态中,直到另一个函数被调用。一个自动CEvent对象在至少一个线程被释放后自动返回一个无标记(无用的)状态。

要使用一个CEvent对象,应在需要时构造一个CEvent对象。指定要等待的事件,应用应拥有它,就可以在构造函数返回时访问事件。调用SetEvent标记(使可用)事件对象,然后当访问完控制资源时,调用Unlock函数。

另一个使用CEvent对象的方法是添加一个CEvent类型的变量,使之成为希望控制的类的一个数据成员。在控制对象被构造期间,可调用CEvent数据成员的构造函数,它指明时间是否是最初就被标记、需要的事件对象类型、事件名称(如果在进程中要使用)和所希望的安全属性。

CEvent类的构造函数原型如下:

CEvent( 

    BOOL bInitiallyOwn /* = FALSE */ ,    //用来指定事件对象初始状态是否为发信状态(默认值为未发信) 


    BOOL bManualReset /* = FALSE */ ,    //用来指定创建的事件对象是自动事件还是手动事件对象(默认值为自动事件对象) 


    LPCTSTR lpszNAme /* = NULL */ ,        //用来定义事件对象的名称 


    LPSECURITY_ATTRIBUTES lpsaAttribute /* = NULL */         //指向一个LPSECURITY_ATTRIBUTES结构的指针 


)

CEvent类提供的三种方法

SetEvent()       //设置事件为发信状态,并释放其他正在等待的线程 

PulseEvent()    //设置事件为发信状态,并释放其他正在等待的线程,然后把事件设置为未发信状态 


ResetEvent()    //设置事件为未发信状态

1.自动事件对象

如果使用CEvent类构造函数的默认参数值的话,则定义的对象为自动事件对象。初始状态为未发信状态,可以用SetEvent使之变为发信状态,等待线程中的第一个线程恢复运行,但事件对象会随即自动将其变为未发信状态,从而使其他处于等待状态的线程仍然被阻塞。就是说,自动事件对象一次只能启动一个处于等待状态的线程。

示例:一个应用程序,当用户在程序窗口上按下鼠标左键时,会创建和启动两个线程,这两个线程被启动后,各自显示一个信息框,表明线程已被启动,随即被事件对象的Lock函数把线程挂起。当用户在程序窗口按下鼠标右键时,启动另一个线程,在该线程中把事件对象置为“发信”状态,从而启动了第一个被挂起的线程。

1.新建单文档程序;

2.在视图类的实现文件中定义一个全局事件对象:

CEvent eventObj;

3.在视图类的实现文件编写如下线程函数:

UINT MessageThread1(LPVOID pParam) 



    LPTSTR pMessage=_T("Thread1 is started" ); 

    CWnd* pMainWnd=AfxGetMainWnd(); 

    ::MessageBox(pMainWnd->m_hWnd,pMessage,_T("Thread message" ),MB_OK); 

    eventObj.Lock();        //线程1处于等待状态 

    /*-----------------------------------------------------------------*/  

    pMessage=_T("Thread1 is unblocked" ); 

    ::MessageBox(pMainWnd->m_hWnd,pMessage,_T("Thread1 message" ),MB_OK);    //显示线程1解锁后的信息框 


    eventObj.Lock();        //线程1再次处于等待状态 

    /*-----------------------------------------------------------------*/  

    pMessage=_T("Thread1 is unblocked again" ); 

    ::MessageBox(pMainWnd->m_hWnd,pMessage,_T("Thread1 message" ),MB_OK);    //显示线程1解锁后的信息框 


     ; 



UINT MessageThread2(LPVOID pParam) 



    LPTSTR pMessage=_T("Thread2 is started" ); 

    CWnd* pMainWnd=AfxGetMainWnd(); 

    ::MessageBox(pMainWnd->m_hWnd,pMessage,_T("Thread message" ),MB_OK); 

    eventObj.Lock();        //线程2处于等待状态 

    /*-----------------------------------------------------------------*/  

    pMessage=_T("Thread2 is unblocked" ); 

    ::MessageBox(pMainWnd->m_hWnd,pMessage,_T("Thread2 message" ),MB_OK);    //显示线程2解锁后的信息框 


     ; 



UINT MessageThread3(LPVOID pParam) 



    eventObj.SetEvent();        //把事件对象置为发信状态 


     ; 

}

4.视图类的鼠标响应消息如下:

void  CThreadTestView::OnLButtonDown(UINT nFlags, CPoint point) 



    AfxBeginThread(MessageThread1, _T("Thread is started" )); //启动线程1 


    AfxBeginThread(MessageThread2, _T("Thread is started" )); //启动线程2 


    CView::OnLButtonDown(nFlags, point); 



 

void  CThreadTestView::OnRButtonDown(UINT nFlags, CPoint point) 



    AfxBeginThread(MessageThread3, _T("Thread is unblocked" )); //启动线程3 


    CView::OnRButtonDown(nFlags, point); 

}

程序运行结果:

2.手工事件对象

手工事件对象一旦用函数SetEvent设置为“发信”状态,就一直处于有效状态,除非又使用对象的成员函数PulseEvent或ResetEvent把它重新设置为“未发信”状态。所以手工事件对象被用来恢复多个处在等待状态线程的运行。

示例:把上面的例子的事件对象定义为手工事件对象,然后运行该程序。

修改为下面代码:

//把定义事件对象的代码改为 

CEvent eventObj(FALSE,TRUE);

程序运行结果:

事件CEvent的使用 .的更多相关文章

  1. C++模拟C#事件委托机制(一)

    原文来自于http://www.cnblogs.com/netssfy/articles/1652671.html 写了一段时间的C#代码后确实发现C#的事件委托非常好用.于是便想是否在C++中也能如 ...

  2. CEvent,CSemaphore,CCriticalSection,CMutex

    一.用CEvent实现线程同步 事件对象(Event)是最简单的同步对象,它包括有信号和无信号两种状态.在线程访问某一资源之前,也许需要等待某一事件的发生,这时用事件对象最合适.例如,只有在通信端口缓 ...

  3. windows下进程间通信与线程间通信

    进程间通信: 1.文件映射(Memory-Mapped Files) 文件映射(Memory-Mapped Files)能使进程把文件内容当作进程地址区间一块内存那样来对待.因此,进程不必使用文件I/ ...

  4. JNI详解---从不懂到理解

    转载:https://blog.csdn.net/hui12581/article/details/44832651 Chap1:JNI完全手册... 3 Chap2:JNI-百度百科... 11 C ...

  5. 【移动端兼容问题研究】javascript事件机制详解(涉及移动兼容)

    前言 这篇博客有点长,如果你是高手请您读一读,能对其中的一些误点提出来,以免我误人子弟,并且帮助我提高 如果你是javascript菜鸟,建议您好好读一读,真的理解下来会有不一样的收获 在下才疏学浅, ...

  6. C++模拟C#事件委托机制(二)

    原文 来自于http://www.cnblogs.com/netssfy/archive/2010/02/02/1662056.html 为了解决非法地址访问的冲突,首先需要知道发生该错误的原因是什么 ...

  7. 【初窥javascript奥秘之事件机制】论“点透”与“鬼点击”

    前言 最近好好的研究了一番移动设备的点击响应速度,期间不断的被自己坑,最后搞得焦头烂额,就是现在可能还有一些问题,但是过程中感觉自己成长不少, 最后居然感觉对javascript事件机制有了更好的认识 ...

  8. memcached(二)事件模型源码分析

    memcachedd事件模型 在memcachedd中,作者为了专注于缓存的设计,使用了libevent来开发事件模型.memcachedd的时间模型同nginx的类似,拥有一个主进行(master) ...

  9. Windows API——CREATEEVENT——创建事件

    事件是一个允许一个线程在某种情况发生时,唤醒另外一个线程的同步对象.事件告诉线程何时去执行某一给定的任务,从而使多个线程流平滑 CreateEvent是创建windows事件的意思,作用主要用在判断线 ...

随机推荐

  1. 如何使用 highlight.js 高亮代码

    highlight 是一款简单易用的 web 代码高亮插件,可以自动检测编程语言并高亮,兼容各种框架,可以说是十分强大了.下面就简单介绍一下如何使用这款插件. 两种使用方式: 1. 手动选择主题,官网 ...

  2. Spring MVC的DispatcherServlet

    Spring MVC提供了一个名为org.springframework.web.servlet.DispatcherServlet的Selvet充当前端控制器,所有的请求驱动都围绕这个Dispatc ...

  3. Win7-64位+Oracle11.2g+使用PLSQL_Developer 的解决办法

    1)安装Oracle 11g 64位 2)安装32位的Oracle客户端( instantclient-basic-win32-11.2.0.1.0)下载instantclient-basic-win ...

  4. 通过SMTP发送邮件的Python代码

    贴上一段用Python开发的发送邮件程序 #coding=UTF-8 import smtplib from email.mime.text import MIMEText smtp_host=&qu ...

  5. React入门教程

    做前端的人都知道,目前热门前端的框架是 VAR => Vue,Anglur,React. 而如果说最热门的前端框架是谁,毫无悬念是 React React 是由 Facebook 主导开发的一个 ...

  6. Hadoop学习笔记二

    一.设置无密码sudo权限,不用在普通用户和root用户间来回切换 chmod u+w /etc/sudoers vim /etc/sudoers #首行添加如下的内容: hadoop ALL=(ro ...

  7. 【OCR技术系列之四】基于深度学习的文字识别(3755个汉字)

    上一篇提到文字数据集的合成,现在我们手头上已经得到了3755个汉字(一级字库)的印刷体图像数据集,我们可以利用它们进行接下来的3755个汉字的识别系统的搭建.用深度学习做文字识别,用的网络当然是CNN ...

  8. 关于ES6 用箭头函数后的 this 指向问题

    最近写完小程序后, 开始学习React, 因为有编译器, 就直接用ES6 新语法了, 中间自然离不开  () => { console.log('箭头函数的this是指向哪的问题')}; var ...

  9. python学习:字典排序

    按字典值排序   按照字典value排序,类似sort -k 命令   import operator x= {1:2,3:4,4:3,2:1,0:0} sorted_x = sorted(x.ite ...

  10. JavaScript 中的对象深度复制(Object Deep Clone)

    JavaScript中并没有直接提供对象复制(Object Clone)的方法. JavaScript中的赋值,其实并不是复制对象,而是类似`c/c++`中的引用(或指针),因此下面的代码中改变对象b ...