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. android典型监听事件实

    public class MainActivity extends Activity { int counter; Button add, sub; TextView display; @Overri ...

  2. mysql插入大量数据

    创建实验表: CREATE TABLE `a` ( `id` int(11) NOT NULL AUTO_INCREMENT, `name` char(50)  NOT NULL, `type` ch ...

  3. maven01 hello maven

    安装省略,注意jdk的版本1.7: 目录:

  4. CSS样式做圆角

    我处理圆角的版本是由内置的绝对定位的四个div组成,每个div都有唯一的圆角图片作CSS Sprite操作.我们将会这样做:  是什么方式导致这项技术表现得这么了不起呢(What makes this ...

  5. mysql 重启

    /etc/init.d/mysql restart /etc/init.d/mysql stop /etc/init.d/mysql start

  6. 在Linux上使用cmake创建CodeBlocks工程

    最近在linux上使用cmake,对于使用GUI习惯的还真不能适应,真是想尽一切办法把原来使用cmake的工程创建成CodeBlocks工程.工程小了还能接受,工程大了太麻烦了. 看了一下cmake的 ...

  7. 在 WinForm 中打开页面采用POST方式传参http。可以多个参数传递,返回json字符串

    //调用方法 Dictionary<string, string> postData = new Dictionary<string, string>(); postData. ...

  8. 模拟app上商品详情点击图片放大并且可以切换大图

    主要使用swiper插件,这里使用各小技巧,就是用两个swiper容器,点击一个小图容器,去让大图容器展示出来 小图容器 <div class="q_banner"> ...

  9. linux 生产环境搭建

    Linux基础命令杂记   今天又一次搞Linux生产环境搭建.这是种步骤很多,很繁琐而且又不得不做的事情.虽然做过很多次,但还是有很多步骤.命令不记得,每一次到处找资料很麻烦,于是将一些步骤记下,以 ...

  10. python 中去除BOM头

    在window的环境下,保存的文本文档会加上三个字符0xEF 0xBB 0xBF的头部,这三个字符可能会影响对文本的读取,形成乱码,在这里记录下如何避免. 首先发现直接保存为ASCII的文本文件是不包 ...