关于事件
  事件(Event)是WIN32提供的最灵活的线程间同步方式,事件可以处于激发状态(signaled or true)或未激发状态(unsignal or false)。根据状态变迁方式的不同,事件可分为两类:
  (1)手动设置:这种对象只可能用程序手动设置,在需要该事件或者事件发生时,采用SetEvent及ResetEvent来进行设置。
  (2)自动恢复:一旦事件发生并被处理后,自动恢复到没有事件状态,不需要再次设置。
  

创建事件的函数原型为:

HANDLE CreateEvent(
 LPSECURITY_ATTRIBUTES lpEventAttributes,
 // SECURITY_ATTRIBUTES结构指针,可为NULL
 BOOL bManualReset, 
 // 手动/自动
 // TRUE:在WaitForSingleObject后必须手动调用ResetEvent清除信号
 // FALSE:在WaitForSingleObject后,系统自动清除事件信号
 BOOL bInitialState, //初始状态
 LPCTSTR lpName //事件的名称
);

使用"事件"机制应注意以下事项:
  (1)如果跨进程访问事件,必须对事件命名,在对事件命名的时候,要注意不要与系统命名空间中的其它全局命名对象冲突;
  (2)事件是否要自动恢复;
  (3)事件的初始状态设置。

看下面代码:

 // event.cpp : 定义控制台应用程序的入口点。
// #include "stdafx.h"
#include <wtypes.h>
#include <iostream>
using namespace std; DWORD WINAPI ThreadProc(LPVOID lpParam);
DWORD WINAPI ThreadProc2(LPVOID lpParam); DWORD g_dwThreadID;
DWORD g_dwThreadID2; UINT g_nTickets = ; //int g_nTickets = 300; //备注1 HANDLE g_hEvent1 = NULL;
HANDLE g_hEvent2 = NULL; CRITICAL_SECTION g_cs; int ThreadCout = ; int _tmain(int argc, _TCHAR* argv[])
{
cout << "Main thread is running." << endl; InitializeCriticalSection(&g_cs);//初始化临界区 HANDLE hHandle = CreateThread(NULL, , ThreadProc, NULL, , &g_dwThreadID);
ThreadCout++;
HANDLE hHandle2 = CreateThread(NULL, , ThreadProc2, NULL, , &g_dwThreadID2);
ThreadCout++; g_hEvent1 = CreateEvent(NULL, FALSE, TRUE, NULL); //备注5:g_hEvent1 = CreateEvent(NULL, TRUE, TRUE, NULL);
g_hEvent2 = CreateEvent(NULL, FALSE, TRUE, NULL); //备注5:g_hEvent2 = CreateEvent(NULL, TRUE, TRUE, NULL); ResetEvent(g_hEvent1);
ResetEvent(g_hEvent2); SetEvent(g_hEvent1); while (TRUE)
{
EnterCriticalSection(&g_cs);
int nCount = ThreadCout;
LeaveCriticalSection(&g_cs); if (nCount == )
{
cout << "Main thread is break." << endl;
break;
} } Sleep(); //备注4 CloseHandle(hHandle);
CloseHandle(hHandle2); DeleteCriticalSection(&g_cs); cout << "Main thread is end." << endl; system("pause");
return ;
} DWORD WINAPI ThreadProc(LPVOID lpParam)
{
// cout << "No." << g_dwThreadID << " thread is running." << endl;
while (TRUE)
{
WaitForSingleObject(g_hEvent1, INFINITE);
cout << "No.1 " << g_dwThreadID << " thread is running." << endl; EnterCriticalSection(&g_cs);
int temp = g_nTickets;
LeaveCriticalSection(&g_cs); cout << "No.1 " << g_dwThreadID << " thread is temp." << endl; if (temp > )
{
Sleep(); //Sleep(1000) //备注2
cout << "No.1-" << g_dwThreadID << " sell ticket : " << temp << endl; EnterCriticalSection(&g_cs);
g_nTickets--;
LeaveCriticalSection(&g_cs); SetEvent(g_hEvent2);
//ResetEvent(g_hEvent1);//备注6
}
else
{
cout << "No.1- break" << endl;
//ResetEvent(g_hEvent1);//备注6
SetEvent(g_hEvent2);//没有这个ThreadProc2不能终止 //备注3
break;
}
} EnterCriticalSection(&g_cs);
ThreadCout--;
LeaveCriticalSection(&g_cs);
cout << "No.1- end" << endl; return ;
} DWORD WINAPI ThreadProc2(LPVOID lpParam)
{
//
while (TRUE)
{
WaitForSingleObject(g_hEvent2, INFINITE);
cout << "No.2 " << g_dwThreadID2 << " thread is running." << endl; EnterCriticalSection(&g_cs);
int temp = g_nTickets;
LeaveCriticalSection(&g_cs); if (temp > )
{
Sleep(); //Sleep(1000) //备注2
cout << "No.2-" << g_dwThreadID2 << " sell ticket : " << temp << endl; EnterCriticalSection(&g_cs);
g_nTickets--;
LeaveCriticalSection(&g_cs); SetEvent(g_hEvent1);
//ResetEvent(g_hEvent2);//备注6
}
else
{
cout << "No.2- break" << endl;
//ResetEvent(g_hEvent2);//备注6
SetEvent(g_hEvent1);//同样的问题,没有这个ThreadProc不能终止 //备注3
break;
}
} EnterCriticalSection(&g_cs);
ThreadCout--;
LeaveCriticalSection(&g_cs); cout << "No.2- end" << endl;
return ;
}

这个代码是接上一遍关于UINT类型作为循环变量的不确定性问题继续完善的,加入了临界区控制全局变量的访问。

本文要说明的是SetEvent和ResetEvent的使用,这个要看备注5和备注6。

备注5处:

CreateEvent的第二个参数决定了是否需要手动调用ResetEvent,当为TRUE时,是需要手动调用,如果不调用,会怎么样呢?不调用,事件会处于一直有信号状态,即备注6处。当为FALSE时候,不需要手动调用,调用不调用,效果一样。把ResetEvent放在WaitForSingleObject前面也是很好的做法。

转载请注明原创链接:http://blog.csdn.net/wujunokay/article/details/12272581

最新文章

  1. datagrid与webAPI的数据交互(ef mvc )
  2. PAT线性结构_一元多项式求导、按给定步长反转链表、出栈序列存在性判断
  3. java8特性深入解读文章合集
  4. 嵌入式 Linux下编译并使用curl静态库
  5. WordPress 主题开发 - (三) 开发工具 待翻译
  6. Codeforces Round #224 (Div. 2)
  7. 用MATLAB画函数的曲线
  8. Java 内省机制
  9. DOM操作HTML文档
  10. HDU 1535 Invitation Cards (POJ 1511)
  11. SVN提交忽略*.class、.classpath、.mymetadata、.project、.settings、.myeclipse和其他非版本控制文件
  12. 纯代码实现CSS圆角
  13. C#中的ArrayList
  14. PHP二维数组合并,根据某个key合并
  15. css常用文本属性
  16. String和StringBuffer的区别?
  17. kubernets code-generator
  18. 如何在 Linux 中查看可用的网络接口
  19. django数据查询之聚合查询和分组查询
  20. sugarCRM文档翻译1

热门文章

  1. poj 1184
  2. C#轻量级日志监控器EasyLogMonitor
  3. reveal 使用注意事项
  4. ES6 js中const,var,let区别 今天第一次遇到const定义的变量
  5. Android APK反编译详解(附图) (转)
  6. GraphQL入门3(Mutation)
  7. minipad2
  8. grep 多行 正则匹配
  9. MySql之存储过程的使用
  10. C#:文件夹匹配