简介:

  在一般情况下,创建一个线程是不能提高程序的执行效率的,所以要创建多个线程。但是多个线程同时运行的时候可能调用线程函数,在多个线程同时对同一个内存地址进行写入,

由于CPU时间调度上的问题,写入数据会被多次的覆盖,所以就要使线程同步。

  同步就是协同步调,按预定的先后次序进行运行。如:你说完,我再说。

  “同”字从字面上容易理解为一起动作 , 其实不是,“同”字应是指协同、协助、互相配合。
  如进程、线程同步,可理解为进程或线程A和B一块配合,A执行到一定程度时要依靠B的某个结果,于是停下来,示意B运行;B依言执行,再将结果给A;A再继续操作。
  在程序中使用多线程时,一般很少有多个线程能在其生命期内进行完全独立的操作。更多的情况是一些线程进行某些处理操作,而其他的线程必须对其处理结果进行了解。
正常情况下对这种处理结果的了解应当在其处理任务完成后进行。
  如果不采取适当的措施,其他线程往往会在线程处理任务结束前就去访问处理结果,这就很有可能得到有关处理结果的错误了解。例如,多个线程同时访问同一个全局变量,
如果都是读取操作,则不会出现问题。如果一个线程负责改变此变量的值,而其他线程负责同时读取变量内容,则不能保证读取到的数据是经过写线程修改后的。
  为了确保读线程读取到的是经过修改的变量,就必须在向变量写入数据时禁止其他线程对其的任何访问,直至赋值过程结束后再解除对其他线程的访问限制。
这种保证线程能了解其他线程任务处理结束后的处理结果而采取的保护措施即为线程同步。
 
  下面实例演示了两个线程同时对一个局部变量进行加操作,多线程资源访问冲突的情况。
  1. #include <Windows.h>
  2. #include <iostream>
  3.  
  4. using namespace std;
  5.  
  6. int number = ;
  7.  
  8. unsigned long __stdcall ThreadProc1(void* lp)
  9. {
  10. while(number < )
  11. {
  12. cout << "thread 1: " << number << endl;
  13. ++number;
  14. _sleep();
  15. }
  16. return ;
  17. }
  18.  
  19. unsigned long __stdcall ThreadProc2(void* lp)
  20. {
  21. while(number < )
  22. {
  23. cout << "thread 1: " << number << endl;
  24. ++number;
  25. _sleep();
  26. }
  27. return ;
  28. }
  29.  
  30. int main()
  31. {
  32. CreateThread(NULL,,ThreadProc1,NULL,,NULL);
  33. CreateThread(NULL,,ThreadProc2,NULL,,NULL);
  34.  
  35. Sleep(*);
  36.  
  37. system("pause");
  38. return ;
  39. }

可以看到有时两个线程计算的值相同,不是我们想要的结果。
 
在WIN32中,同步机制主要有以下几种:
(1)信号量(semaphore);
(2)互斥量(mutex);
(3)临界区(Critical section)
(4)事件(Event);
 
-------------------------------------信号量----------------------------------------------------
 

信号量是维护0到指定最大值之间的同步对象。信号量状态在其计数大于0时是有信号的,而其计数是0时是无信号的。信号量对象在控制上可以支持有限数量共享资源的访问。

信号量的特点和用途可用下列几句话定义:
(1)如果当前资源的数量大于0,则信号量有效;
(2)如果当前资源数量是0,则信号量无效;
(3)系统决不允许当前资源的数量为负值;
(4)当前资源数量决不能大于最大资源数量。

创建信号量

  1. HANDLE CreateSemaphore (
  2.  PSECURITY_ATTRIBUTE psa, //信号量的安全属性
  3.   LONG lInitialCount, //开始时可供使用的资源数
  4.   LONG lMaximumCount, //最大资源数
  5. PCTSTR pszName); //信号量的名称

释放信号量

通过调用ReleaseSemaphore函数,线程就能够对信标的当前资源数量进行递增,该函数原型为:

  1. BOOL WINAPI ReleaseSemaphore(
  2.  HANDLE hSemaphore, //要增加的信号量句柄
  3.  LONG lReleaseCount, //信号量的当前资源数增加lReleaseCount
  4.  LPLONG lpPreviousCount //增加前的数值返回
  5. );

打开信号量

和其他核心对象一样,信号量也可以通过名字跨进程访问,打开信号量的API为:

  1. HANDLE OpenSemaphore (
  2.  DWORD fdwAccess, //access
  3.  BOOL bInherithandle, //如果允许子进程继承句柄,则设为TRUE
  4.  PCTSTR pszName //指定要打开的对象的名字
  5. );

实例

  1. #include<windows.h>
  2. #include<iostream>
  3. using namespace std;
  4.  
  5. int number = ; //定义全局变量
  6. HANDLE hSemaphore; //定义信号量句柄
  7.  
  8. unsigned long __stdcall ThreadProc1(void* lp)
  9. {
  10. long count;
  11. while (number < )
  12. {
  13. WaitForSingleObject(hSemaphore, INFINITE); //等待信号量为有信号状态
  14. cout << "thread 1 :"<<number << endl;
  15. ++number;
  16. _sleep();
  17. ReleaseSemaphore(hSemaphore, , &count);
  18. }
  19.  
  20. return ;
  21. }
  22.  
  23. unsigned long __stdcall ThreadProc2(void* lp)
  24. {
  25. long count;
  26. while (number < )
  27. {
  28. WaitForSingleObject(hSemaphore, INFINITE); //等待信号量为有信号状态
  29. cout << "thread 2 :"<<number << endl;
  30. ++number;
  31. _sleep();
  32. ReleaseSemaphore(hSemaphore, , &count);
  33. }
  34.  
  35. return ;
  36. }
  37.  
  38. int main()
  39. {
  40. hSemaphore = CreateSemaphore(NULL, , , "sema");
  41.  
  42. CreateThread(NULL, , ThreadProc1, NULL, , NULL);
  43. CreateThread(NULL, , ThreadProc2, NULL, , NULL);
  44.  
  45. Sleep(*);
  46.  
  47. system("pause");
  48. return ;
  49. }

-------------------------------------互斥量----------------------------------------------------

采用互斥对象机制。 只有拥有互斥对象的线程才有访问公共资源的权限,因为互斥对象只有一个,所以能保证公共资源不会同时被多个线程访问。互斥不仅能实现同一应用程序的公共资源安全共享,还能实现不同应用程序的公共资源安全共享。

  1. #include<windows.h>
  2. #include<iostream>
  3. using namespace std;
  4.  
  5. int number = ; //定义全局变量
  6. HANDLE hMutex; //定义互斥对象句柄
  7.  
  8. unsigned long __stdcall ThreadProc1(void* lp)
  9. {
  10. while (number < )
  11. {
  12. WaitForSingleObject(hMutex, INFINITE);
  13. cout << "thread 1 :"<<number << endl;
  14. ++number;
  15. _sleep();
  16. ReleaseMutex(hMutex);
  17. }
  18.  
  19. return ;
  20. }
  21.  
  22. unsigned long __stdcall ThreadProc2(void* lp)
  23. {
  24. while (number < )
  25. {
  26. WaitForSingleObject(hMutex, INFINITE);
  27. cout << "thread 2 :"<<number << endl;
  28. ++number;
  29. _sleep();
  30. ReleaseMutex(hMutex);
  31. }
  32.  
  33. return ;
  34. }
  35.  
  36. int main()
  37. {
  38. hMutex = CreateMutex(NULL, false, "mutex"); //创建互斥对象
  39.  
  40. CreateThread(NULL, , ThreadProc1, NULL, , NULL);
  41. CreateThread(NULL, , ThreadProc2, NULL, , NULL);
  42.  
  43. Sleep(*);
  44.  
  45. system("pause");
  46. return ;
  47. }

-------------------------------------临界区----------------------------------------------------

  临界区(Critical Section)是一段独占对某些共享资源访问的代码,在任意时刻只允许一个线程对共享资源进行访问。如果有多个线程试图同时访问临界区,那么在有一个线程进入后其他所有试图访问此临界区的线程将被挂起,并一直持续到进入临界区的线程离开。临界区在被释放后,其他线程可以继续抢占,并以此达到用原子方式操作共享资源的目的。

  临界区在使用时以CRITICAL_SECTION结构对象保护共享资源,并分别用EnterCriticalSection()和LeaveCriticalSection()函数去标识和释放一个临界区。所用到的CRITICAL_SECTION结构对象必须经过InitializeCriticalSection()的初始化后才能使用,而且必须确保所有线程中的任何试图访问此共享资源的代码都处在此临界区的保护之下。否则临界区将不会起到应有的作用,共享资源依然有被破坏的可能。

  1. #include<windows.h>
  2. #include<iostream>
  3. using namespace std;
  4.  
  5. int number = ; //定义全局变量
  6. CRITICAL_SECTION Critical; //定义临界区句柄
  7.  
  8. unsigned long __stdcall ThreadProc1(void* lp)
  9. {
  10. while (number < )
  11. {
  12. EnterCriticalSection(&Critical);
  13. cout << "thread 1 :"<<number << endl;
  14. ++number;
  15. _sleep();
  16. LeaveCriticalSection(&Critical);
  17. }
  18.  
  19. return ;
  20. }
  21.  
  22. unsigned long __stdcall ThreadProc2(void* lp)
  23. {
  24. while (number < )
  25. {
  26. EnterCriticalSection(&Critical);
  27. cout << "thread 2 :"<<number << endl;
  28. ++number;
  29. _sleep();
  30. LeaveCriticalSection(&Critical);
  31. }
  32.  
  33. return ;
  34. }
  35.  
  36. int main()
  37. {
  38. InitializeCriticalSection(&Critical); //初始化临界区对象
  39.  
  40. CreateThread(NULL, , ThreadProc1, NULL, , NULL);
  41. CreateThread(NULL, , ThreadProc2, NULL, , NULL);
  42.  
  43. Sleep(*);
  44.  
  45. system("pause");
  46. return ;
  47. }
 
-------------------------------------事件-实例------------------------------------------------
  1. #include <Windows.h>
  2. #include <iostream>
  3.  
  4. using namespace std;
  5.  
  6. int number = ; //定义全局变量
  7. HANDLE hEvent; //定义事件句柄
  8.  
  9. unsigned long __stdcall ThreadProc1(void* lp)
  10. {
  11. while (number < )
  12. {
  13. WaitForSingleObject(hEvent,INFINITE);
  14. cout << "Thread 1 :" << number << endl;
  15. ++number;
  16. _sleep();
  17. SetEvent(hEvent);
  18. }
  19. return ;
  20. }
  21.  
  22. unsigned long __stdcall ThreadProc2(void* lp)
  23. {
  24. while (number < )
  25. {
  26. WaitForSingleObject(hEvent,INFINITE);
  27. cout << "Thread 2 :" << number << endl;
  28. ++number;
  29. _sleep();
  30. SetEvent(hEvent);
  31. }
  32. return ;
  33. }
  34.  
  35. int main()
  36. {
  37. CreateThread(NULL, , ThreadProc1, NULL, , NULL);
  38. CreateThread(NULL, , ThreadProc2, NULL, , NULL);
  39. hEvent = CreateEvent(NULL, FALSE, TRUE, L"event"); //
  40.  
  41. Sleep(*);
  42.  
  43. system("pause");
  44. return ;
  45. }
 

C++线程同步 -- windows的更多相关文章

  1. 线程同步(windows平台):信号量

    一:介绍 信号量也是系统核心对象,它允许多个线程同一时刻访问同一资源,但需限制同一时刻访问资源的最大线程数目. 信号量遵循规则:1.当前资源计数大于0,信号量有效.2.当前资源计数等于0,信号量无效. ...

  2. 线程同步(windows平台):事件

    一:介绍 事件Event实际上是个内核对象,事件分两种状态:激发状态和未激发状态.分两种类型:手动处置事件和自动处置事件.手动处置事件被设置为激发状态后,会唤醒所有等待的线程,一直保持为激发状态,直到 ...

  3. 线程同步(windows平台):互斥对象

    一:介绍 互斥对象是系统内核维护的一种数据结构,保证了对象对单个线程的访问权. 二:函数说明 创建互斥对象:    HANDLE CreateMutex(            LPSECURITY_ ...

  4. 线程同步(windows平台):临界区

    一:介绍 临界区指的是一个访问共用资源(例:全局变量)的程序片段,该共用资源无法同时被多个线程访问的特性.有多个线程试图同时访问临界区,那么在有一个线程进入后其他所有试图访问此临界区的线程将被挂起,并 ...

  5. windows核心编程 - 线程同步机制

    线程同步机制 常用的线程同步机制有很多种,主要分为用户模式和内核对象两类:其中 用户模式包括:原子操作.关键代码段 内核对象包括:时间内核对象(Event).等待定时器内核对象(WaitableTim ...

  6. windows核心编程---第八章 使用内核对象进行线程同步

    使用内核对象进行线程同步. 前面我们介绍了用户模式下线程同步的几种方式.在用户模式下进行线程同步的最大好处就是速度非常快.因此当需要使用线程同步时用户模式下的线程同步是首选. 但是用户模式下的线程同步 ...

  7. windows核心编程---第七章 用户模式下的线程同步

    用户模式下的线程同步 系统中的线程必须访问系统资源,如堆.串口.文件.窗口以及其他资源.如果一个线程独占了对某个资源的访问,其他线程就无法完成工作.我们也必须限制线程在任何时刻都能访问任何资源.比如在 ...

  8. 重新想象 Windows 8 Store Apps (46) - 多线程之线程同步: Lock, Monitor, Interlocked, Mutex, ReaderWriterLock

    [源码下载] 重新想象 Windows 8 Store Apps (46) - 多线程之线程同步: Lock, Monitor, Interlocked, Mutex, ReaderWriterLoc ...

  9. 重新想象 Windows 8 Store Apps (47) - 多线程之线程同步: Semaphore, CountdownEvent, Barrier, ManualResetEvent, AutoResetEvent

    [源码下载] 重新想象 Windows 8 Store Apps (47) - 多线程之线程同步: Semaphore, CountdownEvent, Barrier, ManualResetEve ...

随机推荐

  1. 阶段3 2.Spring_04.Spring的常用注解_2 常用IOC注解按照作用分类

    注解配置和xml的配置要实现的功能都是一样的.都是要降低程序间的耦合,只不过配置的形式不一样 打包方式改成jar 加入spring 的依赖 复制之前的代码过来 复制到我们新建的工程里 resurces ...

  2. dubbo的启动方法

    Dubbo服务的运行方式: 1.使用Servlet容器运行(Tomcat.Jetty等)----不可取 缺点:增加复杂性(端口.管理) 浪费资源(内存)2.自建Main方法类来运行(spring容器) ...

  3. Day04:继承的意义(下)

    对象转型 向上造型 什么是向上造型? 子类对象赋给父类引用. 父类引用指向子类对象. 子类转成父类 默认进行(父类引用指向子类对象). 为什么需要向上造型? 子类对象可以放入父类类型的数组中. 父类数 ...

  4. Matlab学习笔记0—课程导入

    0,Matlab语言的介绍 1.什么叫计算? 在汉语中,“计算”一词的含义: 谋划 ,考虑 , 算计.随着电子计算机的产生与应用,人们对“计算”的理解发生了很大的变化.             (1) ...

  5. 应用安全 - 中间件 - IIS - 漏洞 - 汇总

    简介 支持协议 HTTP HTTP/2 HTTPS FTP FTPS SMTP NNTP等 支持操作系统 NT/2000/XP Professional/Server 2003及后续版本(XP Hom ...

  6. CF1187E Tree Painting【换根dp】

    题目传送门 题意 一棵$N$个节点的树,初始时所有的节点都是白色,第一次可以选择任意一个把它涂成黑色.接下来,只能把与黑色节点原来相连的白色节点涂成黑色(涂成黑色的点视为被删去,与其它节点不相连).每 ...

  7. (转)Maven的pom.xml文件配置使用

    转载:http://www.cnblogs.com/GarfieldTom/p/3707160.html <project xmlns="http://maven.apache.org ...

  8. python 三元运算、列表推倒式、字典推倒式、生成器生成式

    1.三元运算 name=input('姓名>>: ') res='SB' if name == 'alex' else 'NB' print(res) 2.列表推倒式 #1.示例 egg_ ...

  9. 【挣扎失败】2019CSP-S 游记

    妈耶…… 今年是作为高中生参赛的第一年……然而…… 心痛 洛谷评分,一橙一蓝两紫两黑 我个菜鸡瑟瑟发抖 在考完后三天的信息课码的,刚写没几个字就要下课了 抽空把这个写完

  10. C++中的类型识别

    1,为什么会提出类型识别概念呢? 1,为什么在 C 语言中没有提出这个概念呢,就是因为在 C++ 中引入了面向对象的特性,面向对象里面有一个非常重要的原则就是赋值兼容性原则: 2,在面向对象中可能出现 ...