WaitForSingleObject、WaitForMulitpleObjects和SignalObjectAndWait线程同步

2010-12-10 15:47 类型:生活点滴

用户模式的线程同步机制效率高,如果需要考虑线程同步问题,应该首先考虑用户模式的线程同步方法。但是,用户模式的线程同步有限制,对于多个进程之间的线程同步,用户模式的线程同步方法无能为力。这时,只能考虑使用内核模式。

Windows提供了许多内核对象来实现线程的同步。对于线程同步而言,这些内核对象有两个非常重要的状态:“已通知”状态,“未通知”状态(也有翻译为:受信状态,未受信状态)。Windows提供了几种内核对象可以处于已通知状态和未通知状态:进程、线程、作业、文件、控制台输入/输出/错误流、事件、等待定时器、信号量、互斥对象。

你可以通知一个内核对象,使之处于“已通知状态”,然后让其他等待在该内核对象上的线程继续执行。你可以使用Windows提供的API函数,等待函数来等待某一个或某些内核对象变为已通知状态。

你可以使用WaitForSingleObject函数来等待一个内核对象变为已通知状态:

DWORD WaitForSingleObject(

HANDLE hObject, //指明一个内核对象的句柄

DWORD dwMilliseconds); //等待时间

该函数需要传递一个内核对象句柄,该句柄标识一个内核对象,如果该内核对象处于未通知状态,则该函数导致线程进入阻塞状态;如果该内核对象处于已通知状态,则该函数立即返回WAIT_OBJECT_0。第二个参数指明了需要等待的时间(毫秒),可以传递INFINITE指明要无限期等待下去,如果第二个参数为0,那么函数就测试同步对象的状态并立即返回。如果等待超时,该函数返回WAIT_TIMEOUT。如果该函数失败,返回WAIT_FAILED。可以通过下面的代码来判断:

DWORD dw = WaitForSingleObject(hProcess, 5000); //等待一个进程结束

switch (dw)

{

case WAIT_OBJECT_0:

// hProcess所代表的进程在5秒内结束

break;

case WAIT_TIMEOUT:

// 等待时间超过5秒

break;

case WAIT_FAILED:

// 函数调用失败,比如传递了一个无效的句柄

break;

}

还可以使用WaitForMulitpleObjects函数来等待多个内核对象变为已通知状态:

DWORD WaitForMultipleObjects(

DWORD dwCount, //等待的内核对象个数

CONST HANDLE* phObjects, //一个存放被等待的内核对象句柄的数组

BOOL bWaitAll, //是否等到所有内核对象为已通知状态后才返回

DWORD dwMilliseconds); //等待时间

该函数的第一个参数指明等待的内核对象的个数,可以是0到MAXIMUM_WAIT_OBJECTS(64)中的一个值。phObjects参数是一个存放等待的内核对象句柄的数组。bWaitAll参数如果为TRUE,则只有当等待的所有内核对象为已通知状态时函数才返回,如果为FALSE,则只要一个内核对象为已通知状态,则该函数返回。第四个参数和WaitForSingleObject中的dwMilliseconds参数类似。

该函数失败,返回WAIT_FAILED;如果超时,返回WAIT_TIMEOUT;如果bWaitAll参数为TRUE,函数成功则返回WAIT_OBJECT_0,如果bWaitAll为FALSE,函数成功则返回值指明是哪个内核对象收到通知。

可以如下使用该函数:

HANDLE h[3]; //句柄数组

//三个进程句柄

h[0] = hProcess1;

h[1] = hProcess2;

h[2] = hProcess3;

DWORD dw = WaitForMultipleObjects(3, h, FALSE, 5000); //等待3个进程结束

switch (dw)

{

case WAIT_FAILED:

// 函数呼叫失败

break;

case WAIT_TIMEOUT:

// 超时

break;

case WAIT_OBJECT_0 + 0:

// h[0](hProcess1)所代表的进程结束

break;

case WAIT_OBJECT_0 + 1:

// h[1](hProcess2)所代表的进程结束

break;

case WAIT_OBJECT_0 + 2:

// h[2](hProcess3)所代表的进程结束

break;

}

你也可以同时通知一个内核对象,同时等待另一个内核对象,这两个操作以原子的方式进行:

DWORD SignalObjectAndWait(

HANDLE hObjectToSignal, //通知的内核对象

HANDLE hObjectToWaitOn, //等待的内核对象

DWORD dwMilliseconds, //等待的时间

BOOL bAlertable); //与IO完成端口有关的参数,暂不讨论

该函数在内部使得hObjectToSignal参数所指明的内核对象变成已通知状态,同时等待hObjectToWaitOn参数所代表的内核对象。dwMilliseconds参数的用法与WaitForSingleObject函数类似。

该函数返回如下:WAIT_OBJECT_0,WAIT_TIMEOUT,WAIT_FAILED,WAIT_IO_COMPLETION。

等你需要通知一个互斥内核对象并等待一个事件内核对象的时候,可以这么写:

ReleaseMutex(hMutex);

WaitForSingleObject(hEvent, INFINITE);

可是,这样的代码不是以原子的方式来操纵这两个内核对象。因此,可以更改如下:

SignalObjectAndWait(hMutex, hEvent, INFINITE, FALSE);

来自 “ ITPUB博客 ” ,链接:http://blog.itpub.net/25897606/viewspace-704361/,如需转载,请注明出处,否则将追究法律责任。

linux 线程 waitforsingleobject,WaitForSingleObject、WaitForMulitpleObjects和SignalObjectAndWait线程同步...相关推荐

  1. linux线程间通信优点,进程间通信与线程间通信【转】

    一个进程写管道:写入字节数小于PIPE_BUF是原子操作,写操作在管道缓冲区没有及时读走时发生阻塞. 一个进程读管道:读操作在管道缓冲区没有数据时发生阻塞. 以前一直想找个机会总结一下进程和线程的通信 ...

  2. 事件EVENT,WaitForSingleObject(),WaitForMultipleObjecct()和SignalObjectAndWait() 的使用(下)

    注意:当WaitForMultipleObjects等待多个内核对象的时候,如果它的bWaitAll 参数设置为false.其返回值减去WAIT_OBJECT_0 就是参数lpHandles数组的序号 ...

  3. 事件EVENT,WaitForSingleObject(),WaitForMultipleObjecct()和SignalObjectAndWait() 的使用(上)

    用户模式的线程同步机制效率高,如果需要考虑线程同步问题,应该首先考虑用户模式的线程同步方法.但是,用户模式的线程同步有限制,对于多个进程之间的线程同步,用户模式的线程同步方法无能为力.这时,只能考虑使 ...

  4. Java线上问题排障:Linux内核bug引发JVM死锁导致线程假死

    Java本质上还是离不开操作系统,一来Java源码是用C/C++实现的,二来java进程还是需要依附于操作系统和硬件资源,有时候一些问题是操作系统级别导致的,下面的整个事件是源自一则真实的线上案例. ...

  5. linux线程怎样实时性,高效轻型线程怎么提高Linux实时性能?

    这些用户空间扩展(有多个)已首先由电信/网络高性能 IP 数据包处理系统进行驱动,以实现所谓的"裸金属"实施,其中,多核设备中的Linux用户空间应用可以模拟"无操作系统 ...

  6. 【Linux 内核】进程管理 ( 内核线程概念 | 内核线程、普通进程、用户线程 | 内核线程与普通进程区别 | 内核线程主要用途 | 内核线程创建函数 kernel_thread 源码 )

    文章目录 一.内核线程概念 二.内核线程.普通进程.用户线程 三.内核线程.普通进程区别 四.内核线程主要用途 五.内核线程创建函数 kernel_thread 源码 一.内核线程概念 直接 由 Li ...

  7. linux创建线程未定义,Linux中未定义的对p线程_CREATE的引用

    Linux中未定义的对p线程_CREATE的引用#include #include #define NUM_THREADS     5void *PrintHello(void *threadid){ ...

  8. Linux 平台 C/C++ 代码中设置线程名

    一般来说,Linux 平台的 C/C++ 程序可以用 prctl() 或 pthreads 的 pthread_setname_np() 接口为一个线程设置线程名.prctl() 可以用于为当前线程设 ...

  9. linux clone线程,如何在Linux上使用clone()创建真正的线程?

    我正在尝试使用clone()创建一个新线程.使用以下代码(-): #include #include #include #define _SCHED_H 1 #define __USE_GNU 1 # ...

最新文章

  1. 2022-2028年中国无滴消雾大棚膜行业市场研究及前瞻分析报告
  2. 可怕!315 曝光 50 多款App“窃听”:这条黑色产业链,有人靠你的隐私年赚千万...
  3. linux find命令的日常使用
  4. Python PIL ImageDraw 和ImageFont模块学习
  5. 体感(Kinect)技术开发和应用简介
  6. 如何使用通用Mapper
  7. 2017年10月23日23:58:04
  8. 深入理解Java虚拟机——JVM类加载机制(类加载过程和类加载器)
  9. Git 上传文件到 码云 gitee
  10. 常见形式 Web API 的简单分类总结
  11. 使用.net Stopwatch class 来分析你的代码
  12. 2015年《大数据》高被引论文Top10文章No.2——大数据时代的数据挖掘 —— 从应用的角度看大数据挖掘(上)...
  13. PS VR发售临近,索尼的VR影视内容也不远了
  14. 硬核黑科技、技术大咖、AI 音乐节……科大讯飞全球 1024 开发者节太燃了!
  15. mysql可以登陆sqlyog1862_MySQL错误号码1862:your password has expired
  16. 云课堂智慧职教计算机基础答案,云课堂智慧职教题库答案护理系内科,智慧职教职业生涯规划答案,智慧职教mooc学院计算机文化基础答案...
  17. 微信8.0表情没有特效怎么回事
  18. pcb元器件通孔焊盘激光焊锡的优势
  19. Mixly第26课~第28课,内容抢先看 | Mixly米思齐纯干货系列
  20. POJ 2954-Triangle(计算几何+皮克定理)

热门文章

  1. Developer Express右键菜单显示汉化
  2. ebay如何确定同一电脑登陆了多个账号,以及同一账号登陆过多台电脑?
  3. 如何使用BorgBackup,Rclone和Wasabi云存储推出自己的备份解决方案
  4. raspberry pi_Raspberry Pi支持的杂耍性能
  5. google开源数学引擎_Google的开源PDF引擎,开放无线运动等
  6. (44)常用终端命令总结
  7. Bootstrap CSS 编码规范之Less 和 Sass 中的操作符
  8. 四元数与复数之间的关系
  9. mysql教程qt linux_一步步学Qt,第四天-Linux 下mysql数据库链接
  10. 子查询返回多个字段_ElasticSearch搜索之QueryFiltering、多/单字符串的多字段查询...