CreateEvent函数详解参见本博客文章:

c++中CreateEvent函数解析(2)

HANDLE CreateEvent(
LPSECURITY_ATTRIBUTES lpEventAttributes,
BOOL bManualReset, 
BOOL bInitialState,
LPCSTR lpName
);
bManualReset:TRUE,使用ResetEvent()手动重置为无信号状态;FALSE,当一个等待线程被释放时,自动重置状态为无信号状态。

bInitialState:指定事件对象的初始状态,当TRUE,初始状态为有信号状态;当FALSE,初始状态为无信号状态。

下面主要演示一下采用CreateEvent实现多线程。

例子很简单,主要测试CreateEvent中bManualReset:和bInitialState参数的取值在线程调用中信号状态的情况。

测试1:

bManualReset:TRUE
bInitialState:TRUE

CreateEvent(NULL, TRUE, TRUE, NULL); //使用手动重置为无信号状态,初始化时有信号状态

example.cpp

#include "iostream"
#include "windows.h"
using namespace std;DWORD WINAPI ThreadProc1(LPVOID lpParam);
DWORD WINAPI ThreadProc2(LPVOID lpParam);
HANDLE hEvent = NULL;
HANDLE hThread1 = NULL;
HANDLE hThread2 = NULL;
int main(int argc, char *args[])
{hEvent = CreateEvent(NULL, TRUE, TRUE, NULL); //使用手动重置为无信号状态,初始化时有信号状态//hEvent = CreateEvent(NULL, FALSE, TRUE, NULL); //当一个等待线程被释放时,自动重置为无信号状态,初始是有信号状态//if (SetEvent(hEvent))//{//    cout << "setEvent 成功" <<endl;//}hThread1 = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)ThreadProc1, NULL, 0,NULL);Sleep(200);hThread2 = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)ThreadProc2, NULL, 0,NULL);Sleep(200);if ( NULL == hThread1){cout <<"create thread fail!";}//DWORD dCount = ResumeThread(hThread);//cout << LOWORD(dCount) << endl;return 0;
}
DWORD WINAPI ThreadProc1(LPVOID lpParam)
{cout <<"in thread1@!"<<endl;DWORD dReturn = WaitForSingleObject(hEvent,INFINITE);if ( WAIT_OBJECT_0 == dReturn){cout <<" thread1 signaled ! "<<endl;}cout <<"in thread1 --signal"<<endl;//SetEvent(hEvent);return 0;
}
DWORD WINAPI ThreadProc2(LPVOID lpParam)
{cout <<"in thread2@!"<<endl;DWORD dReturn = WaitForSingleObject(hEvent,INFINITE);if ( WAIT_OBJECT_0 == dReturn){cout <<"thread2 signaled ! "<<endl;}cout <<"in thread2--signal"<<endl;return 0;
}

执行结果:

从结果中看,执行完线程1又执行了线程2.

由于hEvent = CreateEvent(NULL, TRUE, TRUE, NULL),使用手动重置为无信号状态,初始化时有信号状态

所以hEvent一直处于有信号状态,无论是线程1释放后,hEvent仍处于有信号状态,所以线程2正常执行了。

测试2:

bManualReset:FALSE
bInitialState:TRUE

hEvent = CreateEvent(NULL, FALSE, TRUE, NULL); //当一个等待线程被释放时,自动重置为无信号状态,初始是有信号状态

example2.cpp

#include "iostream"
#include "windows.h"
using namespace std;DWORD WINAPI ThreadProc1(LPVOID lpParam);
DWORD WINAPI ThreadProc2(LPVOID lpParam);
HANDLE hEvent = NULL;
HANDLE hThread1 = NULL;
HANDLE hThread2 = NULL;
int main(int argc, char *args[])
{//hEvent = CreateEvent(NULL, TRUE, TRUE, NULL); //使用手动重置为无信号状态,初始化时有信号状态hEvent = CreateEvent(NULL, FALSE, TRUE, NULL); //当一个等待线程被释放时,自动重置为无信号状态,初始是有信号状态//if (SetEvent(hEvent))//{//    cout << "setEvent 成功" <<endl;//}hThread1 = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)ThreadProc1, NULL, 0,NULL);Sleep(200);hThread2 = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)ThreadProc2, NULL, 0,NULL);Sleep(200);if ( NULL == hThread1){cout <<"create thread fail!";}//DWORD dCount = ResumeThread(hThread);//cout << LOWORD(dCount) << endl;return 0;
}
DWORD WINAPI ThreadProc1(LPVOID lpParam)
{cout <<"in thread1@!"<<endl;DWORD dReturn = WaitForSingleObject(hEvent,INFINITE);if ( WAIT_OBJECT_0 == dReturn){cout <<" thread1 signaled ! "<<endl;}cout <<"in thread1 --signal"<<endl;//SetEvent(hEvent);return 0;
}
DWORD WINAPI ThreadProc2(LPVOID lpParam)
{cout <<"in thread2@!"<<endl;DWORD dReturn = WaitForSingleObject(hEvent,INFINITE);if ( WAIT_OBJECT_0 == dReturn){cout <<"thread2 signaled ! "<<endl;}cout <<"in thread2--signal"<<endl;return 0;
}

执行结果:


从执行结果中分析,执行了线程1,线程2一直在等待,直到主线程结束。

由于hEvent = CreateEvent(NULL, FALSE, TRUE, NULL),当一个等待线程被释放时,自动重置为无信号状态,初始是有信号状态

初始执行线程1的时候,hEvent是有信号的,所以线程1正常执行;又由于bManualReset=FALSE,所以执行完线程1后,hEvent自动重置为无信号状态,所以在线程2中,

WaitForSingleObject(hEvent,INFINITE);

函数一直在等待hEvent变为有信号状态,但是当主线程执行完,还没等待到,线程2程序一直没有走下去。

测试3:

bManualReset:TRUE
bInitialState:FALSE

hEvent = CreateEvent(NULL, TRUE, FALSE, NULL);//使用手动重置为无信号状态,初始化时为无信号状态

example3.cpp

#include "iostream"
#include "windows.h"
using namespace std;DWORD WINAPI ThreadProc1(LPVOID lpParam);
DWORD WINAPI ThreadProc2(LPVOID lpParam);
HANDLE hEvent = NULL;
HANDLE hThread1 = NULL;
HANDLE hThread2 = NULL;
int main(int argc, char *args[])
{//hEvent = CreateEvent(NULL, TRUE, TRUE, NULL); //使用手动重置为无信号状态,初始化时有信号状态//hEvent = CreateEvent(NULL, FALSE, TRUE, NULL); //当一个等待线程被释放时,自动重置为无信号状态,初始是有信号状态hEvent = CreateEvent(NULL, TRUE, FALSE, NULL);//使用手动重置为无信号状态,初始化时为无信号状态//if (SetEvent(hEvent))//{//   cout << "setEvent 成功" <<endl;//}hThread1 = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)ThreadProc1, NULL, 0,NULL);Sleep(200);hThread2 = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)ThreadProc2, NULL, 0,NULL);Sleep(200);if ( NULL == hThread1){cout <<"create thread fail!";}//DWORD dCount = ResumeThread(hThread);//cout << LOWORD(dCount) << endl;return 0;
}
DWORD WINAPI ThreadProc1(LPVOID lpParam)
{cout <<"in thread1@!"<<endl;DWORD dReturn = WaitForSingleObject(hEvent,INFINITE);if ( WAIT_OBJECT_0 == dReturn){cout <<" thread1 signaled ! "<<endl;}cout <<"in thread1 --signal"<<endl;//SetEvent(hEvent);return 0;
}
DWORD WINAPI ThreadProc2(LPVOID lpParam)
{cout <<"in thread2@!"<<endl;DWORD dReturn = WaitForSingleObject(hEvent,INFINITE);if ( WAIT_OBJECT_0 == dReturn){cout <<"thread2 signaled ! "<<endl;}cout <<"in thread2--signal"<<endl;return 0;
}

执行结果,可想而知,只能输出:

in thread1@!
in thread2@!

因为初始为无信号状态,所以hEvent一直处于无信号状态,因此这两个线程一直在等待,直到主线程结束。

修改:放开例子中的注释部分:

if (SetEvent(hEvent))//设置信号为有信号状态
{
cout << "setEvent 成功" <<endl;
}

执行结果:

可见,线程1和线程2都执行了。

因为调用SetEvent,事件变为有信号状态,线程1执行;又由于线程1释放后,hEvent仍旧处于有信号状态,所以线程2也执行了。

再修改:在线程1中,添加ResetEvent(hEvent)(手动设置事件为无信号状态),则线程2不会执行。

测试4:

bManualReset:FALSE
bInitialState:FALSE

hEvent = CreateEvent(NULL, FALSE, FALSE, NULL);//线程释放后自动重置为无信号状态,初始化时为无信号状态

example4.cpp

#include "iostream"
#include "windows.h"
using namespace std;DWORD WINAPI ThreadProc1(LPVOID lpParam);
DWORD WINAPI ThreadProc2(LPVOID lpParam);
HANDLE hEvent = NULL;
HANDLE hThread1 = NULL;
HANDLE hThread2 = NULL;
int main(int argc, char *args[])
{//hEvent = CreateEvent(NULL, TRUE, TRUE, NULL); //使用手动重置为无信号状态,初始化时有信号状态//hEvent = CreateEvent(NULL, FALSE, TRUE, NULL); //当一个等待线程被释放时,自动重置为无信号状态,初始是有信号状态//hEvent = CreateEvent(NULL, TRUE, FALSE, NULL);//使用手动重置为无信号状态,初始化时为无信号状态hEvent = CreateEvent(NULL, FALSE, FALSE, NULL);//使用手动重置为无信号状态,初始化时为无信号状态if (SetEvent(hEvent)){cout << "setEvent 成功" <<endl;}hThread1 = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)ThreadProc1, NULL, 0,NULL);Sleep(200);hThread2 = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)ThreadProc2, NULL, 0,NULL);Sleep(200);if ( NULL == hThread1){cout <<"create thread fail!";}//DWORD dCount = ResumeThread(hThread);//cout << LOWORD(dCount) << endl;return 0;
}
DWORD WINAPI ThreadProc1(LPVOID lpParam)
{cout <<"in thread1@!"<<endl;DWORD dReturn = WaitForSingleObject(hEvent,INFINITE);if ( WAIT_OBJECT_0 == dReturn){cout <<" thread1 signaled ! "<<endl;}cout <<"in thread1 --signal"<<endl;//SetEvent(hEvent);return 0;
}
DWORD WINAPI ThreadProc2(LPVOID lpParam)
{cout <<"in thread2@!"<<endl;DWORD dReturn = WaitForSingleObject(hEvent,INFINITE);if ( WAIT_OBJECT_0 == dReturn){cout <<"thread2 signaled ! "<<endl;}cout <<"in thread2--signal"<<endl;return 0;
}

执行结果:

由于调用SetEvent,hEvent为有信号状态,线程1正常执行,又由于调用完线程1后,hEvent自动重置为无信号状态,所以线程2只能在等待,直到主线程退出。

修改:线程1中的SetEvent(hEvent);的注释去掉,再运行,则线程1和线程2 都会执行。

c++CreateEvent函数在多线程中使用及实例相关推荐

  1. sleep 函数 linux 线程吗,sleep函数在多线程中的作用

    1.前言 多线程中经常会使用sleep()函数,我们知道cpu对于多线程的操作是采用时间片轮询的方式,即,时间片1操作线程A,时间片1结束后,时间片2操作线程B,时间片2结束后,时间片3操作线程A,依 ...

  2. java的start()函数是什么意思_在Java中,start()函数在多线程中做了什么? - Break易站...

    Java 多线程 在这两种方法中,我们都覆盖了run()函数,但是我们通过调用start()函数来启动一个线程.那么为什么我们不直接调用oveerridden run()函数呢?为什么总是调用star ...

  3. java线程休眠sleep函数_Java多线程中sleep()方法详解及面试题

    一. Java线程生命周期(五个阶段) 新建状态就绪状态运行状态阻塞状态死亡状态 如图 二.sleep方法 API中的解释 static voidsleep(long millis) 使当前正在执行的 ...

  4. c++中CreateEvent函数解析(1)

    函数原型: HANDLE CreateEvent(LPSECURITY_ATTRIBUTES lpEventAttributes, // SDBOOL bManualReset, // reset t ...

  5. c++中CreateEvent函数

    http://blog.csdn.net/chenyujing1234/article/details/8572921 函数原型: [cpp]  view plain copy HANDLE Crea ...

  6. python中run函数作用_python3多线程中如何改写run()函数?

    我们对于函数的使用一般是直接根据其作用进行举例讲解,最近偶然的一次多线程的代码练习中,让小编发现在构建多线程的时候,我们也可以对函数进行重写.小编马上进行了这个函数内容的整理,为了让大家能理解前后的内 ...

  7. c++中CreateEvent函数解析(2)

    函数原型: HANDLE CreateEvent(LPSECURITY_ATTRIBUTES lpEventAttributes, // SD BOOL bManualReset, // reset ...

  8. 【Linux】多线程中使用fork()

    (最核心的东西我在下面用红色字体标出来了,理解了那块,这些东西都是就理解了!) 在本篇文章开始之前,需要大家先了解线程和进程,这位大哥讲的言简意赅:进程和线程的主要区别(总结)_kuangsongha ...

  9. android串口补位,Rust多线程中的消息传递机制

    代码说话. use std::thread; use std::sync::mpsc; use std::time::Duration; fn main() { let (tx, rx) = mpsc ...

最新文章

  1. 「小程序JAVA实战」小程序的举报功能开发(68)
  2. JAVA常见面试题之Error、RuntimeException、CheCkedException
  3. [云炬创业管理笔记]第一章讨论1
  4. Java多线程和并发(三),Thread类和Runnable接口
  5. 今天写的上传类,纯练手之作,供新人学习
  6. 前端构建工具gulp入门教程
  7. 圆形渐变shader_Flutter 中渐变的高级用法
  8. Canvas绘图基础(一)
  9. Rust 碎碎念:【译】Arc 在 Rust 中是如何工作的
  10. python中文编码(汉字乱码问题解决方案)
  11. Qt + 运动控制 (固高运动控制卡)【1】环境准备,框架搭建
  12. 两个VB程序之交换数据的DDE工程
  13. iOS小技能:金额格式处理 (货币符号本地化)
  14. [点评] [电子科技大学][TCP/IP协议原理][杨宁]
  15. 记忆翻牌游戏——react算法学习
  16. QT常用函数和总结(持续更新)
  17. 静态创意和动态创意_2020年创意工作的5个预测
  18. Java操作Excel基础--POI之HSSF
  19. 招行信用卡中心2020开发笔试
  20. 移远 BC35-G 模块固件升级

热门文章

  1. sed 首行加一行,末尾加一行
  2. android h文件是什么文件,Android 资源文件
  3. 移动距离 奇妙的数字
  4. LocalSolver-全领域、超大规模混合变量数学规划介绍
  5. 扫一扫,一键生成微信个人数据报告
  6. WIN7 64位操作系统下成功安装Oracle10g(32位)
  7. 翻译视频字幕的软件叫什么?安利这几个软件给你
  8. 财务和计算机操作方面的专业知识,会计基础的学习方法
  9. 1.计算机二级考试 计算机基础知识部分
  10. linux安装svn使用解压包的方式