事件CEvent的使用 .

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解锁后的信息框 

    return  0 ; 

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解锁后的信息框 

    return  0 ; 

UINT MessageThread3(LPVOID pParam) 

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

    return  0 ; 
}

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);

程序运行结果:

时间: 2024-10-04 04:08:38

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

运用多线程技术实现文件的快速搜索

从一名初学者到如今一名熟练的VC程序员,得益于VC知识库的帮助,一直想着该如何回报.现撰此文,一来略表心意,二来和VC爱好者交流. 操作系统一般都提供了文件搜索的功能,但采用的是顺序搜索,搜索效率很底.而且按此法编程十分烦琐,在目录层次很多时,往往不好处理.本文采用多线程技术实现文件的快速搜索,代码量很少,执行效率极高. 其基本思想其实很简单,就是找到一个目录就开辟一个线程,文件的话当然在线程内就处理了.这样实现了同步搜索,速度当然快起来了. 本文程序运行效果图 以下介绍VC具体实现: 一. 搜

VC下利用多线程实现文件的快速检索

操作系统一般都提供了文件搜索的功能,但采用的是顺序搜索,搜索效率很低.而且按此法编程十分烦琐,在目录层次很多时,往往不好处理.本文采用多线程技术实现文件的快速搜索,代码量很少,执行效率极高. 本文提出的多线程文件搜索的基本思想很简单,就是找到一个目录就开辟一个线程,而找到一个文件当然在线程内就处理了,这样实现了同步搜索. 以下介绍其在vc平台下的具体实现: 1.搜索用到两个win32的两个函数 HANDLE FindFirstFile(LPCTSTR lpFileName,LPWIN32_FIN

iOS最新面试锦集

1. 为什么说Objective-C是一门动态的语言? ① 什么是动态语言? 动态语言,是指程序在运行时可以改变其结构:新的函数可以被引进,已有的函数可以被删除等在结构上的变化.比如众所周知的ECMAScript(JavaScript)便是一个动态语言.除此之外如Ruby.Python等也都属于动态语言,而C.C++等语言则不属于动态语言. 有三个名词容易混淆: Dynamic Programming Language (动态语言或动态编程语言) Dynamically Typed Langua

线程通信--利用事件对象

在线程之间传送信号可以使用事件对象,用MFC的CEvent来表示.一个事件对象有两种状态:信号态与非信号态.线程能监视于信号态的事件,以便在适当的时间完成对事件的操作. 创建事件的语句如下: CEvent ThreadBegin; 事件被创建后就自动处于非信号态,要使用它处于信息态,必须使用对件对象的成员函数SetEvent(),即: ThreadBegin.SetEvent(); 线程监视这个信号来知道事件已准备就绪,从而可以进行其它的操作.而线程通过调用API函数WaitForSingleO

Yii框架组件和事件行为管理详解_php实例

本文实例讲述了Yii框架组件和事件行为管理.分享给大家供大家参考,具体如下: Yii是一个基于组件.用于开发大型 Web 应用的高性能 PHP 框架.CComponent几乎是所有类的基类,它控制着组件与事件的管理,其方法与属性如下,私有变量$_e数据存放事件(evnet,有些地方叫hook),$_m数组存放行为(behavior). 组件管理 YII是一个纯oop框架,很多类中的成员变量的受保护或者私有的,CComponent中利用php中的魔术方法__get(),__set()来访问和设置属

深入解析PHP的Yii框架中的event事件机制_php技巧

事件事件可以将自定义代码"注入"到现有代码中的特定执行点.附加自定义代码到某个事件,当这个事件被触发时,这些代码就会自动执行.例如,邮件程序对象成功发出消息时可触发 messageSent 事件.如想追踪成功发送的消息,可以附加相应追踪代码到messageSent 事件. Yii 引入了名为 yii\base\Component 的基类以支持事件.如果一个类需要触发事件就应该继承 yii\base\Component 或其子类. Yii的event机制YII的事件机制,是其比较独特之处

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

在memcachedd中,作者为了专注于缓存的设计,使用了libevent来开发事件模型.memcachedd的时间模型同nginx的类似,拥有一个主进行(master)以及多个工作者线程(woker). 流程图 在memcached中,是先对工作者线程进行初始化并启动,然后才会创建启动主线程. 工作者线程 初始化 memcached对工作者线程进行初始化,参数分别为线程数量以及`main_base`, /* start up worker threads if MT mode */ threa

Yii框架组件和事件行为管理详解

本文实例讲述了Yii框架组件和事件行为管理.分享给大家供大家参考,具体如下: Yii是一个基于组件.用于开发大型 Web 应用的高性能 PHP 框架.CComponent几乎是所有类的基类,它控制着组件与事件的管理,其方法与属性如下,私有变量$_e数据存放事件(evnet,有些地方叫hook),$_m数组存放行为(behavior). 组件管理 YII是一个纯oop框架,很多类中的成员变量的受保护或者私有的,CComponent中利用php中的魔术方法__get(),__set()来访问和设置属

<转>关于MFC的多线程类 CSemaphore,CMutex,CCriticalSection,CEvent

---------------------------------------------------线程---------------------------------------------------- CreateThread(NULL,0,(LPTHREAD_START_ROUTINE) listen,NULL,0,&IDThread); 参数1 : 按MSDN上的说法,为空时handle不可以被继承. 参数2 : 栈的大小, 为0时取默认值. 参数3 : 线程函数指针, 上面的li