1.  互斥量,Mutex

  1. #include <Windows.h>
  2. #include <iostream>
  3. using namespace std;
  4. DWORD WINAPI Thread1(LPVOID lpParmeter);
  5. DWORD WINAPI Thread2(LPVOID lpParmeter);
  6. static HANDLE g_hMutex = INVALID_HANDLE_VALUE;
  7. static int g_iCnt = 100;
  8. int main()
  9. {
  10. HANDLE hThread1 = INVALID_HANDLE_VALUE;
  11. HANDLE hThread2 = INVALID_HANDLE_VALUE;
  12. g_hMutex = CreateMutex(NULL, FALSE, "Mutex");
  13. // 第二个参数:创建者是否拥有所有权,FALSE为没有所有权,
  14. // 遇到第一个WaitForSingleObject的时候就把所有权给它,
  15. // 所以Thread1里面的WaitForSingleObject(g_hMutex, INFINITE)能够继续执行
  16. if (!g_hMutex)
  17. {
  18. cout << "Failed to CreateMutex !" << endl;
  19. return 0;
  20. }
  21. hThread1 = CreateThread(NULL, 0, Thread1, NULL, 0, NULL);
  22. hThread2 = CreateThread(NULL, 0, Thread2, NULL, 0, NULL);
  23. Sleep(4000);        // 让2个线程有足够的时间执行完操作。
  24. CloseHandle(hThread1);
  25. CloseHandle(hThread2);
  26. system("PAUSE");
  27. return 0;
  28. }
  29. DWORD WINAPI Thread1(LPVOID lpParmeter)
  30. {
  31. while (true)
  32. {
  33. // 请求事件对象
  34. WaitForSingleObject(g_hMutex, INFINITE);    // INFINITE: 长时间等待,差不多50天左右吧!
  35. if (g_iCnt > 0)
  36. {
  37. Sleep(20);
  38. cout << "Thread1:" << g_iCnt-- << endl;
  39. ReleaseMutex(g_hMutex);                 // 释放资源
  40. }
  41. else
  42. {
  43. ReleaseMutex(g_hMutex);
  44. break;
  45. }
  46. }
  47. return 0;
  48. }
  49. DWORD WINAPI Thread2(LPVOID lpParameter)//thread data
  50. {
  51. while (true)
  52. {
  53. // 请求事件对象
  54. WaitForSingleObject(g_hMutex,INFINITE);
  55. if (g_iCnt > 0)
  56. {
  57. Sleep(20);
  58. cout << "thread2:" << g_iCnt-- << endl;
  59. ReleaseMutex(g_hMutex);
  60. }
  61. else
  62. {
  63. ReleaseMutex(g_hMutex);
  64. break;
  65. }
  66. }
  67. return 0;
  68. }

几个注意的地方:

(1)互斥量为内核对象,能够与其他线程或特殊事件取得同步;

(2)速度比临界区要慢;

(3)互斥量对象与所有其它内核对象的不同之处在于它是被线程所拥有的,互斥量对象除了记录当前信号状态外,还要记住此时那个线程拥有它。

(4)这个常来被运用于限制程序启动次数!

2.事件 Event

  1. #include <Windows.h>
  2. #include <iostream>
  3. using namespace std;
  4. DWORD WINAPI Thread1(LPVOID lpParmeter);
  5. DWORD WINAPI Thread2(LPVOID lpParmeter);
  6. static HANDLE g_hEvent = INVALID_HANDLE_VALUE;
  7. static int g_iCnt = 100;
  8. int main()
  9. {
  10. HANDLE hThread1 = INVALID_HANDLE_VALUE;
  11. HANDLE hThread2 = INVALID_HANDLE_VALUE;
  12. g_hEvent = CreateEvent(NULL, false, false, "Event");
  13. if (!g_hEvent)
  14. {
  15. cout << "Failed to CreateEvent !" << endl;
  16. return 0;
  17. }
  18. /*HANDLE CreateEvent(
  19. LPSECURITY_ATTRIBUTES lpEventAttributes,    // SECURITY_ATTRIBUTES结构指针,可为NULL
  20. BOOL bManualReset,                          // 手动/自动
  21.                                             // TRUE: 在WaitForSingleObject后必须手动调用ResetEvent清除信号
  22.     // FALSE:在WaitForSingleObject后,系统自动清除事件信号
  23. BOOL bInitialState,                         // 初始状态
  24. LPCTSTR lpName                              // 事件的名称
  25. );*/
  26. hThread1 = CreateThread(NULL, 0, Thread1, NULL, 0, NULL);
  27. hThread2 = CreateThread(NULL, 0, Thread2, NULL, 0, NULL);
  28. SetEvent(g_hEvent);
  29. Sleep(4000);        // 让2个线程有足够的时间执行完操作。
  30. CloseHandle(hThread1);
  31. CloseHandle(hThread2);
  32. system("PAUSE");
  33. return 0;
  34. }
  35. DWORD WINAPI Thread1(LPVOID lpParmeter)
  36. {
  37. while (true)
  38. {
  39. // 请求事件对象
  40. WaitForSingleObject(g_hEvent, INFINITE);    // INFINITE: 长时间等待,差不多50天左右吧!
  41. if (g_iCnt > 0)
  42. {
  43. Sleep(20);
  44. cout << "Thread1:" << g_iCnt-- << endl;
  45. SetEvent(g_hEvent);
  46. }
  47. else
  48. {
  49. SetEvent(g_hEvent);
  50. break;
  51. }
  52. }
  53. return 0;
  54. }
  55. DWORD WINAPI Thread2(LPVOID lpParameter)//thread data
  56. {
  57. while (true)
  58. {
  59. // 请求事件对象
  60. WaitForSingleObject(g_hEvent,INFINITE);
  61. if (g_iCnt > 0)
  62. {
  63. Sleep(20);
  64. cout << "thread2:" << g_iCnt-- << endl;
  65. SetEvent(g_hEvent);
  66. }
  67. else
  68. {
  69. SetEvent(g_hEvent);
  70. break;
  71. }
  72. }
  73. return 0;
  74. }

几个注意的地方:

(1).和Mutex使用差不多,只有细微的差别;

(2).可以使用SetEvent或ResetEvent改变其状态;

(3).在应用程序中任意一处没有正确的按照规则调用SetEvent或ResetEvent,将达不到同步或互斥的目的;

(4).一般来说,都是利用Event来进行同步,而不是我们这里的让它来达到互斥;

(5).Event处于无信号状态时,相关线程或进程退出,系统并不会尝试将其置为有信号状态;

3.临界区 CRITICAL_SECTION

  1. #include <Windows.h>
  2. #include <iostream>
  3. using namespace std;
  4. DWORD WINAPI Thread1(LPVOID lpParmeter);
  5. DWORD WINAPI Thread2(LPVOID lpParmeter);
  6. CRITICAL_SECTION g_CriticalSection; // 定义
  7. static int g_iCnt = 100;
  8. int main()
  9. {
  10. HANDLE hThread1 = INVALID_HANDLE_VALUE;
  11. HANDLE hThread2 = INVALID_HANDLE_VALUE;
  12. InitializeCriticalSection(&g_CriticalSection);      // 初始化
  13. hThread1 = CreateThread(NULL, 0, Thread1, NULL, 0, NULL);
  14. hThread2 = CreateThread(NULL, 0, Thread2, NULL, 0, NULL);
  15. Sleep(4000);        // 让2个线程有足够的时间执行完操作。
  16. CloseHandle(hThread1);
  17. CloseHandle(hThread2);
  18. DeleteCriticalSection(&g_CriticalSection);          // 删除
  19. system("PAUSE");
  20. return 0;
  21. }
  22. DWORD WINAPI Thread1(LPVOID lpParmeter)
  23. {
  24. while (true)
  25. {
  26. EnterCriticalSection(&g_CriticalSection);       // 进入临界区,获得所有权,其他线程就等待
  27. if (g_iCnt > 0)
  28. {
  29. Sleep(20);
  30. cout << "Thread1:" << g_iCnt-- << endl;
  31. LeaveCriticalSection(&g_CriticalSection);   // 离开临界区,释放所有权
  32. }
  33. else
  34. {
  35. LeaveCriticalSection(&g_CriticalSection);
  36. break;
  37. }
  38. }
  39. return 0;
  40. }
  41. DWORD WINAPI Thread2(LPVOID lpParameter)//thread data
  42. {
  43. while (true)
  44. {
  45. EnterCriticalSection(&g_CriticalSection);
  46. if (g_iCnt > 0)
  47. {
  48. Sleep(20);
  49. cout << "thread2:" << g_iCnt-- << endl;
  50. LeaveCriticalSection(&g_CriticalSection);
  51. }
  52. else
  53. {
  54. LeaveCriticalSection(&g_CriticalSection);
  55. break;
  56. }
  57. }
  58. return 0;
  59. }

几个注意的地方:

(1).比Mutex速度快;

(2).临界区在线程内的分配必须是全局的;

(3). 临界区一次只允许一个线程访问;

4.信号量Semaphore

首先说说那些关于信号量那些不得不让人伤心的事情,因为笔者大学不好学习,非常调皮,而信号量又是老师最讨论及考试的话题之一,所以我觉得这个东西非常扯淡,非常影响情绪,于是放在最后。------以上是为题外话。

为什么大学老师总是喜欢信号量呢?

因为这是一个生产者-消费者模型,并且很多计算机问题都可以看做是生产者-消费者的问题,是同步最易理解的模型。

关于理论上的知识,我就不说了,书里面很多的。

还有我不是很想实现生产者-消费者的模型,就用其他例子代替了。这个有点不负责任。

  1. #include <Windows.h>
  2. #include <iostream>
  3. #include <vector>
  4. using namespace std;
  5. DWORD WINAPI Thread1(LPVOID lpParmeter);
  6. DWORD WINAPI Thread2(LPVOID lpParmeter);
  7. static HANDLE g_hSemaphore = INVALID_HANDLE_VALUE;; // 定义
  8. static int g_iCnt = 100;
  9. int main()
  10. {
  11. HANDLE hThread1 = INVALID_HANDLE_VALUE;
  12. HANDLE hThread2 = INVALID_HANDLE_VALUE;
  13. // HANDLE CreateSemaphore (
  14. //  PSECURITY_ATTRIBUTE psa,
  15. //  LONG lInitialCount, //开始时可供使用的资源数
  16. //  LONG lMaximumCount, //最大资源数
  17. //  PCTSTR pszName);
  18. g_hSemaphore = CreateSemaphore(NULL, 1, 1, "Semaphore");
  19. // 初始化有1个信号量。
  20. if (g_hSemaphore == INVALID_HANDLE_VALUE)
  21. {
  22. cout << "Failed to Create Semaphore!" << endl;
  23. return 0;
  24. }
  25. hThread1 = CreateThread(NULL, 0, Thread1, NULL, 0, NULL);
  26. hThread2 = CreateThread(NULL, 0, Thread2, NULL, 0, NULL);
  27. Sleep(4000);        // 让2个线程有足够的时间执行完操作。
  28. CloseHandle(hThread1);
  29. CloseHandle(hThread2);
  30. system("PAUSE");
  31. return 0;
  32. }
  33. DWORD WINAPI Thread1(LPVOID lpParmeter)
  34. {
  35. while (true)
  36. {
  37. WaitForSingleObject(g_hSemaphore, INFINITE);        // 释放一个信号量
  38. if (g_iCnt > 0)
  39. {
  40. Sleep(20);
  41. cout << "Thread1:" << g_iCnt-- << endl;
  42. ReleaseSemaphore(g_hSemaphore, 1, NULL);        // 增加一个信号量
  43. }
  44. else
  45. {
  46. break;
  47. }
  48. }
  49. return 0;
  50. }
  51. DWORD WINAPI Thread2(LPVOID lpParameter)//thread data
  52. {
  53. while (true)
  54. {
  55. WaitForSingleObject(g_hSemaphore, INFINITE);
  56. if (g_iCnt > 0)
  57. {
  58. Sleep(20);
  59. cout << "thread2:" << g_iCnt-- << endl;
  60. ReleaseSemaphore(g_hSemaphore, 1, NULL);
  61. }
  62. else
  63. {
  64. break;
  65. }
  66. }
  67. return 0;
  68. }

几个注意的地方:

信号量内核对象对线程的同步方式与前面几种不同,它允许多个线程在同一时刻访问某一资源,但是需要限制同一时刻访问此资源的最大线程数目。

总结: 线程规模 = CPU 数 * 2 + 1

Windows下C++多线程同步与互斥简单运用的更多相关文章

  1. Windows下C++多线程同步与互斥简单运用(转)

    1.  互斥量,Mutex #include <Windows.h> #include <iostream> using namespace std; DWORD WINAPI ...

  2. 使用cwRsync实现windows下文件定时同步【转】

    1.参考文献: 使用cwRsync实现windows下文件定时同步(备份) 文件同步工具CwRsync的使用方法及常用命令详解 2.背景: 当前的SCADA架构中,有1台Server,5台FE,还有1 ...

  3. 使用cwRsync实现windows下文件定时同步

    1.参考文献: 使用cwRsync实现windows下文件定时同步(备份) 文件同步工具CwRsync的使用方法及常用命令详解 2.背景: 当前的SCADA架构中,有1台Server,5台FE,还有1 ...

  4. C#Stimulator项目>>>C/C++ DLL的生成和调用,Windows下的多线程

    Windows下的多线程 http://blog.csdn.net/ganpengjin1/article/category/2541791 使用C/C++建立DLL,环境VS2013 新建Win32 ...

  5. Windows下安装Redis服务、搭建简单Redis主从复制

    Redis拥有非常强大的主从复制功能,而且还支持一个master可以拥有多个slave,而一个slave又可以拥有多个slave,从而形成强大的多级服务器集群架构.目前在同一台window下安装三个r ...

  6. Windows 下针对python脚本做一个简单的进程保护

    前提: 大家运行的脚本程序经常会碰到系统异常关闭.或被其他用户错杀的情况.这样就需要一个进程保护的工具. 本文结合windows 的计划任务,实现一个简单的进程保护的功能. 利用py2exe生产 ex ...

  7. Windows下的多线程

    Windows下的进程和Linux下的进程是不一样的,它比较懒惰,从来不执行任何东西,它只是为线程提供执行环境,然后由线程负责执行包含在进程的地址空间中的代码.当创建一个进程的时候,操作系统会自动创建 ...

  8. windows 下的 Rsync 同步

    整理一下 windows 下的 rsync 文件同步. Rsync下载地址: 链接:https://pan.baidu.com/s/1nL0Ee_u76ytWKUFMeiKDIw 提取码:52in 一 ...

  9. Windows下PHP多线程扩展pthreads的安装

    pthreads扩展安装步骤 1.查看phpinfo() 获取PHP版本号及位数(x86表示32位,x64表示64位).编译器版本.PHP配置文件加载所在位置等.如下图所示: 2.pthreads扩展 ...

随机推荐

  1. 7-05. 魔法优惠券(25) (数学 ZJU_PAT)

    题目链接:http://www.patest.cn/contests/ds/7-05 在火星上有个魔法商店,提供魔法优惠券.每一个优惠劵上印有一个整数面值K,表示若你在购买某商品时使用这张优惠劵.能够 ...

  2. 实现了在android实现左右滑动切换界面的效果

    这是实现了在android实现左右滑动切换界面的效果,该效果的源码下载,请到源码天堂下载吧,喜欢的朋友可以研究一下. 布局文件 <?xml version="1.0" enc ...

  3. cocos2d_android 瞬间动作

    该文章所写的瞬间动作主要有CCFlipX,CCFlipY,CCHide,CCShow 当中CCFlipX是以Y轴为中心旋转,CCFlipY是以X轴为中心旋转,CCHide将精灵对象隐藏,CCShow将 ...

  4. 20个热门jQuery的提示和技巧

    以下是一些非常有用的jQuery提示和所有jQuery的开发技巧. 1.优化性能复杂的选择 查询DOM中的一个子集,使用复杂的选择时,大幅提高了性能: var subset = $("&qu ...

  5. Flash Recovery Area空间不足导致DB不能打开或hang住处理方法

    当归档目录设置在闪回恢复区,并且闪回恢复区又满了的情况下, DB 就会无法归档而hang住或者无法打开. 这种情况下打开数据库会遇到如下错误信息: SQL> select status from ...

  6. springmvc jstl

    springmvc运用maven的jetty插件运行成功,部署在tomcat6报错:ClassNotFoundException: javax.servlet.jsp.jstl.core.Config ...

  7. uva11029 - Leading and Trailing

    题目: 求n的k次方,然后将答案用前三位和最后三位表示. Sample Input 2 123456 1 123456 2 Sample Output 123...456 152...936 分析: ...

  8. Sybase Power Designer 16.5破解版下载

    http://pan.baidu.com/s/1ddsjs  下载后正常安装,然后将压缩文件里的dll文件拷到安装目录下覆盖原文件,启动Power Designer后,选择help-->abou ...

  9. uilable 换行标记

    m_tipLabel.lineBreakMode = UILineBreakModeWordWrap; m_tipLabel.numberOfLines = 0; m_tipLabel.text =  ...

  10. BZOJ 3503: [Cqoi2014]和谐矩阵( 高斯消元 )

    偶数个相邻, 以n*m个点为变量, 建立异或方程组然后高斯消元... O((n*m)^3)复杂度看起来好像有点大...但是压一下位的话就是O((n*m)^3 / 64), 常数小, 实际也跑得很快. ...