http://blog.csdn.net/xiaobai1593/article/details/6672193

在多线程下面,有时候我们会希望等待某一线程完成了再继续做其他事情,要实现这个目的,可以使用Windows API函数WaitForSingleObject,或者WaitForMultipleObjects。这两个函数都会等待Object被标为有信号(signaled)时才返回的。
那么,什么是信号呢?
简单来说,Windows下创建的Object都会被赋予一个状态量。如果Object被激活了,或者正在使用,那么该Object就是无信号,也就是不可用;另一方面,如果Object可用了,那么它就恢复有信号了。

这两个函数的优点是它们在等待的过程中会进入一个非常高效沉睡状态,只占用极少的CPU时间片。(这两个函数都是在内核状态下等待内核对象,不切换到用户模式下,因而效率很高)

WaitForSingleObject()

1. 格式

DWORD WaitForSingleObject( HANDLE hHandle, DWORDdwMilliseconds);

有两个参数,分别是THandle和Timeout(毫秒单位)。

如果想要等待一条线程,那么你需要指定线程的Handle,以及相应的Timeout时间。当然,如果你想无限等待下去,Timeout参数可以指定系统常量INFINITE。

2. 使用对象

它可以等待如下几种类型的对象:

Event,Mutex,Semaphore,Process,Thread

3. 返回类型

有三种返回类型:

WAIT_OBJECT_0, 表示等待的对象有信号(对线程来说,表示执行结束);

WAIT_TIMEOUT, 表示等待指定时间内,对象一直没有信号(线程没执行完);

WAIT_ABANDONED 表示对象有信号,但还是不能执行  一般是因为未获取到锁或其他原因

示例:

[cpp] view plaincopy
  1. <span style="font-family:Times New Roman;">#include <windows.h>
  2. #include <stdio.h>
  3. #include <iostream.h>
  4. //声明函数  创建线程
  5. DWORD WINAPI FunProc( LPVOID lpParameter);
  6. void main()
  7. {
  8. HANDLE hThread;
  9. hThread=CreateThread(NULL,0,FunProc,NULL,0,NULL);
  10. DWORD dwRet=WaitForSingleObject(hThread, 1);
  11. if(dwRet==WAIT_OBJECT_0)
  12. {
  13. printf("创建的线程执行结束\n");
  14. }
  15. if(dwRet==WAIT_TIMEOUT)
  16. {
  17. printf("等待超时\n");
  18. }
  19. if(dwRet==WAIT_ABANDONED)
  20. {
  21. printf("Abandoned\n");
  22. }
  23. CloseHandle(hThread);
  24. }
  25. DWORD WINAPI FunProc( LPVOID lpParameter )
  26. {
  27. int i=1;
  28. for(; i<1000; i++)
  29. {
  30. printf("%d  ", i);
  31. if(! (i%10))
  32. printf("\n");
  33. }
  34. return 0;
  35. }</span>

WaitForMultipleObjecct

相对来说,WaitForMultipleObjects要复杂点点

格式为:

DWORD WaitForMultipleObjects(DWORD nCount, CONST HANDLE *lpHandles, BOOLfWaitAll, DWORDdwMilliseconds);

四个参数分别是:

1. nCount,DWORD类型,用于指定句柄数组的数量
2. lphObjects,Pointer类型,用于指定句柄数组的内存地址
3. fWaitAll,Boolean类型,True表示函数等待所有指定句柄的Object有信号为止
4. dwTimeout,DWORD类型,用于指定等待的Timeout时间,单位毫秒,可以是INFINITE

当WaitForMultipleObjects等待多个内核对象的时候,如果它的bWaitAll 参数设置为false。其返回值减去WAIT_OBJECT_0 就是参数lpHandles数组的序号。如果同时有多个内核对象被触发,这个函数返回的只是其中序号最小的那个。

如果为TRUE 则等待所有信号量有效在往下执行。(FALSE 当有其中一个信号量有效时就向下执行)

问题就在这里,我们如何可以获取所有被同时触发的内核对象。

举个例子:我们需要在一个线程中处理从完成端口、数据库、和可等待定时器来的数据。一个典型的实现方法就是:用WaitForMultipleObjects等待所有的这些事件。如果完成端口,数据库发过来的数据量非常大,可等待定时器时间也只有几十毫秒。那么这些事件同时触发的几率可以说非常大,我们不希望丢弃任何一个被触发的事件。那么如何能高效地实现这一处理呢?   

MSDN中有一句非常重要的描述,它可以说是WaitForMultipleObjects用法的精髓:The function modifies the state of some types of synchronization objects. Modification occurs only for the object or objects whose signaled state caused the function to return. For example, the count of a semaphore object is decreased by one. When bWaitAll is FALSE, and multiple objects are in the signaled state, the function chooses one of the objects to satisfy the wait; the states of the objects not selected are unaffected.   

多个内核对象被触发时,WaitForMultipleObjects选择其中序号最小的返回。而WaitForMultipleObjects它只会改变使它返回的那个内核对象的状态。

这儿又会产生一个问题,如果序号最小的那个对象频繁被触发,那么序号比它大的内核对象将得不到被处理的机会。为了解决这一问题,可以采用双WaitForMultipleObjects检测机制来实现。见下面的例子:

[cpp] view plaincopy
  1. <span style="font-family:Times New Roman;">DWORD WINAPI ThreadProc(LPVOID lpParameter)   
  2. {   
  3. DWORD dwRet = 0;   
  4. int nIndex = 0;   
  5. while(1)   
  6. { 
  7. dwRet = WaitForMultipleObjects(nCount,pHandles,false,INFINITE);   
  8. switch(dwRet)   
  9. {
  10. case WAIT_TIMEOUT:   
  11. break;   
  12. case WAIT_FAILED:   
  13. return 1;
  14. default:
  15. {
  16. nIndex = dwRet - WAIT_OBJECT_0;   
  17. ProcessHanlde(nIndex++);   //同时检测其他的事件   
  18. while(nIndex < nCount) //nCount事件对象总数   
  19. {   
  20. dwRet = WaitForMultipleObjects(nCount - nIndex,&pHandles[nIndex],false,0);   
  21. switch(dwRet)   
  22. {
  23. case WAIT_TIMEOUT:   
  24. nIndex = nCount; //退出检测,因为没有被触发的对象了.   
  25. break;
  26. case WAIT_FAILED:   
  27. return 1;   
  28. default:
  29. {
  30. nIndex = dwRet - WAIT_OBJECT_0;   
  31. ProcessHanlde(nIndex++);   
  32. }
  33. break;
  34. }//switch结束
  35. }//while结束
  36. }//default结束
  37. break;
  38. }//switch结束
  39. }//while结束
  40. return 0;
  41. }</span>

MSDN对于这个函数的返回值还有一句话:   

Return Values   If the function succeeds, the return value indicates the event that caused the function to return. This value can be one of the following.   ValueMeaning   WAIT_OBJECT_0 to (WAIT_OBJECT_0 + nCount – 1)If bWaitAll is TRUE, the return value indicates that the state of all specified objects is signaled.   If bWaitAll is FALSE, the return value minus WAIT_OBJECT_0 indicates the lpHandles array index of the object that satisfied the wait. If more than one object became signalled during the call, this is the array index of the signalled object with the smallest index value of all the signalled objects.   WAIT_ABANDONED_0 to (WAIT_ABANDONED_0 + nCount – 1)If bWaitAll is TRUE, the return value indicates that the state of all specified objects is signaled and at least one of the objects is an abandoned mutex object.   If bWaitAll is FALSE, the return value minus WAIT_ABANDONED_0 indicates the lpHandles array index of an abandoned mutex object that satisfied the wait.   WAIT_TIMEOUTThe time-out interval elapsed and the conditions specified by the bWaitAll parameter are not satisfied.   

返回值

如果函数成功,返回值表示该事件导致该函数返回。

这个值可以是下列之一:

WAIT_OBJECT_0到WAIT_OBJECT_0 + nCount - 1

如果bWaitAll为TRUE,则返回值表明所有指定对象的状态信号

如果bWaitAll为FALSE,则返回值减去不是WAIT_OBJECT_0表示lpHandles数组的对象的满意指数的等待。如果多个对象在通话过程中信号成为有信号状态,这是与所有的信号对象的最小索引值的信号对象的数组索引。

WAIT_ABANDONED_0至WAIT_ABANDONED_0 + nCount - 1

如果bWaitAll为TRUE,则返回值表明所有指定对象的状态,至少是暗示的对象之一,是一个废弃的互斥对象。

如果bWaitAll为FALSE,则返回值减去WAIT_ABANDONED_0表示lpHandles数组的一个废弃的互斥对象的满意指数的等待。   

WAIT_TIMEOUTThe超时间隔已过,由bWaitAll参数指定的条件得不到满足。

转:WaitForSingleObject()函数、WaitForMultipleObject()函数相关推荐

  1. 在用户线程/主线程中推荐MsgWaitForMultipleObjects代替WaitForSingleObject和WaitForMultipleObjects()函数

    在多线程编程中,通常都需要线程间的同步,一个线程要等待另一个线程的事件才继续执行,一般的做法是采用WaitForSingleObject和WaitForMultipleObjects()函数来实现. ...

  2. Java箭头函数,lambda函数

    Java箭头函数,lambda函数 lambda表达式实质就是一种语法糖,(建议尽量不要使用lambda表达式,代码太简洁,难懂,难以调试) lambda表达式的简单例子: 不需要参数,返回5 ()- ...

  3. C++ 笔记(13)— 函数(函数声明、函数定义、函数调用[传值、指针、引用]、函数参数默认值、函数重载)

    每个 C++ 程序都至少有一个函数,即主函数 main() ,所有简单的程序都可以定义其他额外的函数. 1. 函数声明 函数声明告诉编译器函数的名称.返回类型和参数.函数声明包括以下几个部分: ret ...

  4. python nums函数_Python函数

    一.简介 函数是可重用的程序代码块.函数的作用,不仅可以实现代码的复用,更能实现代码的一致性.一致性指的是,只要修改函数的代码,则所有调用该函数的地方都能得到体现. 函数用关键字def来定义,def关 ...

  5. Python源码学习:Python函数浅析-函数闭包

    Python源码分析 本文环境python2.5系列 参考书籍<<Python源码剖析>> 上一篇分析了函数参数的分析后,本文分析函数闭包的实现.函数闭包即函数定义和函数表达式 ...

  6. querywrapper or and嵌套_两个经常用的逻辑函数AND函数,OR函数

    在逻辑函数里AND函数,OR函数是使用频率是比较高的函数,但像NOT函数和XOR两个函数就略显鸡肋了. 今天我们重点对前两个函数进行讲述,后两个大家可以自行查看下,知道就行了,等哪天可以用上的时候再进 ...

  7. R语言使用trimws函数:trimws函数去除(删除、remove)字符串头尾的空格

    R语言使用trimws函数:trimws函数去除(删除.remove)字符串头尾的空格 目录 R语言使用trimws函数:trimws函数去除(删除.remove)字符串头尾的空格

  8. R语言使用integrate函数进行函数积分计算实战

    R语言使用integrate函数进行函数积分计算实战 目录 R语言使用integrate函数进行函数积分计算实战 #自定义函数 #积分操作

  9. R语言ggplot2使用geom_line函数geom_point函数可视化哑铃图、并对哑铃图进行排序(reorder dumbbell plot)

    R语言ggplot2使用geom_line函数geom_point函数可视化哑铃图.并对哑铃图进行排序(reorder dumbbell plot) 目录

  10. R语言ggplot2可视化:将dataframe和数据列名称传递给函数通过函数进行ggplot2可视化输出

    R语言ggplot2可视化:将dataframe和数据列名称传递给函数通过函数进行ggplot2可视化输出 目录

最新文章

  1. 为什么edge AI是一个无需大脑的人
  2. android上的i-jetty (1)环境搭建
  3. 网络推广营销浅析网站度过“沙盒期”后,为什么还不收录?
  4. curl get请求_Linux curl 常用示例你都 Get 了吗?| CSDN 博文精选
  5. if else复合语句
  6. paip.提升用户体验---gcc c++ try catch不能捕获异常的解决--使用 SEH Ca6
  7. 福特dms系统服务器,长安福特DMS使用规范.pdf
  8. C# ZPL打印标签
  9. 行人重识别论文阅读11-BDTR 红外光行人重识别
  10. 407. 接雨水 II【我亦无他唯手熟尔】
  11. 系统分析师---论软件的系统测试及应用
  12. linux切换内核版本
  13. SignalCatcher
  14. 《ppt》word插入批注,修改作者
  15. 会java需要多久能学会python_java好学吗?零基础学java要多久?
  16. 【无标题】PMP强化练习正确题一
  17. Zhong__Python reduce()
  18. WordPress数据库的基本构造(转载自http://blog.csdn.net/ppiao1970hank/article/details/6301812)...
  19. 在中国大陆,XGP对比Steam有什么优势?
  20. 物联网下的智能家居 生活智能化不再遥远

热门文章

  1. 个人笔记1:display与visibility用法
  2. 使用 jQuery 避免鼠标双击
  3. Linux系统攻略关于vi替换命令的使用说明
  4. systemtap打点方法
  5. 内核线程、轻量级进程、用户线程三种线程概念解惑(线程≠轻量级进程)
  6. 自己编译操作系统,安装systemtap
  7. Android StrictMode类使用实例
  8. hive get_json_object json_tuple json解析详解
  9. python中对象排序的两种方式
  10. python 正则表达式 \b 大坑