秒杀多线程第四篇 一个经典的多线程同步问题

上一篇《秒杀多线程第三篇原子操作 Interlocked系列函数》中介绍了原子操作在多进程中的作用,现在来个复杂点的。这个问题涉及到线程的同步和互斥,是一道非常有代表性的多线程同步问题,如果能将这个问题搞清楚,那么对多线程同步也就打下了良好的基础。

 

程序描述:

主线程启动10个子线程并将表示子线程序号的变量地址作为参数传递给子线程。子线程接收参数 -> sleep(50) -> 全局变量++ -> sleep(0) -> 输出参数和全局变量。

要求:

1.子线程输出的线程序号不能重复。

2.全局变量的输出必须递增。

下面画了个简单的示意图:

分析下这个问题的考察点,主要考察点有二个:

1.主线程创建子线程并传入一个指向变量地址的指针作参数,由于线程启动须要花费一定的时间,所以在子线程根据这个指针访问并保存数据前,主线程应等待子线程保存完毕后才能改动该参数并启动下一个线程。这涉及到主线程与子线程之间的同步

2.子线程之间会互斥的改动和输出全局变量。要求全局变量的输出必须递增。这涉及到各子线程间的互斥

 

下面列出这个程序的基本框架,可以在此代码基础上进行修改和验证。

[cpp]  view plain copy
  1. //经典线程同步互斥问题  
  2. #include <stdio.h>  
  3. #include <process.h>  
  4. #include <windows.h>  
  5.   
  6. long g_nNum; //全局资源  
  7. unsigned int __stdcall Fun(void *pPM); //线程函数  
  8. const int THREAD_NUM = 10; //子线程个数  
  9.   
  10. int main()  
  11. {  
  12.     g_nNum = 0;  
  13.     HANDLE  handle[THREAD_NUM];  
  14.       
  15.     int i = 0;  
  16.     while (i < THREAD_NUM)   
  17.     {  
  18.         handle[i] = (HANDLE)_beginthreadex(NULL, 0, Fun, &i, 0, NULL);  
  19.         i++;//等子线程接收到参数时主线程可能改变了这个i的值  
  20.     }  
  21.     //保证子线程已全部运行结束  
  22.     WaitForMultipleObjects(THREAD_NUM, handle, TRUE, INFINITE);    
  23.     return 0;  
  24. }  
  25.   
  26. unsigned int __stdcall Fun(void *pPM)  
  27. {  
  28. //由于创建线程是要一定的开销的,所以新线程并不能第一时间执行到这来  
  29.     int nThreadNum = *(int *)pPM; //子线程获取参数  
  30.     Sleep(50);//some work should to do  
  31.     g_nNum++;  //处理全局资源  
  32.     Sleep(0);//some work should to do  
  33.     printf("线程编号为%d  全局资源值为%d\n", nThreadNum, g_nNum);  
  34.     return 0;  
  35. }  

运行结果可以参考下列图示,强烈建议读者亲自试一试。

1

2

3

可以看出,运行结果完全是混乱和不可预知的。本系列将会运用Windows平台下各种手段包括关键段,事件,互斥量,信号量等等来解决这个问题并作一份全面的总结,敬请关注。

 

秒杀多线程第五篇 经典线程同步 关键段CS》已经发布,欢迎参阅。

秒杀多线程第六篇 经典线程同步 事件Event》已经发布,欢迎参阅。

秒杀多线程第七篇 经典线程同步 互斥量Mutex》已经发布,欢迎参阅。

秒杀多线程第八篇 经典线程同步 信号量Semaphore》已经发布,欢迎参阅。 

 

 

转载请标明出处,原文地址:http://blog.csdn.net/morewindows/article/details/7442333

 

73
1
我的同类文章
主题推荐
函数 color 多线程 线程 91
猜你在找
Hadoop 2.X大数据平台基础
Windows Server 2012 R2 远程桌面管理
Windows Server 2012 R2 Hyper-V 管理
Windows Server 2012 Hyper-v 管理
Windows Server 2012 组策略管理
秒杀多线程第四篇 一个经典的多线程同步问题
秒杀多线程第四篇 一个经典的多线程同步问题
秒杀多线程第四篇---一个经典的多线程同步问题
秒杀多线程第四篇 一个经典的多线程同步问题
秒杀多线程第四篇 一个经典的多线程同步问题 httpblogcsdnnetmorewindowsarticledetails7442333
id="cproIframe_u1607657_2" width="728" height="90" src="http://pos.baidu.com/acom?adn=3&at=6&aurl=&cad=1&ccd=24&cec=UTF-8&cfv=0&ch=0&col=zh-CN&conOP=0&cpa=1&dai=2&dis=0&ltr=https%3A%2F%2Fwww.baidu.com%2Flink%3Furl%3DqgiER_kmNYikRcWfLUsdYEWph82ZVX_6Rz9USHJZYbFg4jEZ6ySygRPllGTtwleN2Ew-6b9LumqJn6LIhauw2wM7yGIPHRjToFk9K9udRSW%26wd%3D%26eqid%3Df29ae40e000b0a7a00000002567b4f49&ltu=http%3A%2F%2Fblog.csdn.net%2Fmorewindows%2Farticle%2Fdetails%2F7442333&lunum=6&n=csdnbaidu_cpr&pcs=1349x643&pis=10000x10000&ps=4414x405&psr=1366x768&pss=1349x4432&qn=47970ebc0c813fa6&rad=&rsi0=728&rsi1=90&rsi5=4&rss0=%23FFFFFF&rss1=%23FFFFFF&rss2=%230000FF&rss3=%23444444&rss4=%23008000&rss5=&rss6=%23e10900&rss7=&scale=&skin=&td_id=1607657&tn=text_default_728_90&tpr=1450921986748&ts=1&version=2.0&xuanting=0&exps=110212&dtm=BAIDU_DUP2_SETJSONADSLOT&dc=2&di=u1607657&ti=%E7%A7%92%E6%9D%80%E5%A4%9A%E7%BA%BF%E7%A8%8B%E7%AC%AC%E5%9B%9B%E7%AF%87%20%E4%B8%80%E4%B8%AA%E7%BB%8F%E5%85%B8%E7%9A%84%E5%A4%9A%E7%BA%BF%E7%A8%8B%E5%90%8C%E6%AD%A5%E9%97%AE%E9%A2%98%20-%20MoreWindows%20Blog%20-%20%E5%8D%9A%E5%AE%A2%E9%A2%91%E9%81%93%20-%20CSDN.NET&tt=1450921986713.280.336.337&feid=110212" align="center,center" marginwidth="0" marginheight="0" scrolling="no" frameborder="0" allowtransparency="true">
查看评论
31楼  yi9un 2015-10-29 20:59发表 [回复]
刚学习线程,29行 int nThreadNum = *(int *)pPM ,怎么理解,为什么nThreadNum会自增,没有 nThreadNum++啊
30楼  vine_branches 2015-07-08 17:29发表 [回复]
[cpp]  view plain copy
  1. #include "stdafx.h"  
  2. #include "stdio.h"  
  3. #include <process.h>  
  4. #include <windows.h>  
  5.   
  6. #define THREAD_NUM  10 // THREAD_NUM > 64会有问题  
  7. volatile long g_nLoginCount;  
  8.   
  9. unsigned int _stdcall ThreadFun4(void * pPM)  
  10. {  
  11.     int nThreadNum = *(int *)pPM;  
  12.     g_nLoginCount++;  
  13.     printf("Thread number is %d,  g_nLoginCount is %d\n", nThreadNum, g_nLoginCount);    
  14.     return 0;  
  15. }  
  16. int _tmain(int argc, _TCHAR* argv[])  
  17. {  
  18.     g_nLoginCount = 0;  
  19.     HANDLE handle[THREAD_NUM];  
  20.       
  21.       for (int i = 0; i < THREAD_NUM; i++)  
  22.     {  
  23.         handle[i] = (HANDLE)_beginthreadex(NULL, 0, ThreadFun4, &i, 0, NULL);  
  24.     }  
  25.     WaitForMultipleObjects(THREAD_NUM, handle, true, INFINITE);  
  26.     return 0;  
  27. }  

以上是我的代码,i值乱序可以理解,为什么nLoginCount 会出现递减的情况呀?
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值