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

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

大家好,又见面了,我是全栈君,祝每个程序员都可以多学几门语言。

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

 

程序描写叙述:

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

要求:

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

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

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

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

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

1.主线程创建子线程并传入一个指向变量地址的指针作參数,因为线程启动需要花费一定的时间,所以在子线程依据这个指针訪问并保存数据前,主线程应等待子线程保存完成后才干修改该參数并启动下一个线程。这涉及到主线程与子线程之间的同步

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

 

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

//经典线程同步相互排斥问题
#include <stdio.h>
#include <process.h>
#include <windows.h>

long g_nNum; //全局资源
unsigned int __stdcall Fun(void *pPM); //线程函数
const int THREAD_NUM = 10; //子线程个数

int main()
{
	g_nNum = 0;
	HANDLE  handle[THREAD_NUM];
	
	int i = 0;
	while (i < THREAD_NUM) 
	{
		handle[i] = (HANDLE)_beginthreadex(NULL, 0, Fun, &i, 0, NULL);
		i++;//等子线程接收到參数时主线程可能改变了这个i的值
	}
	//保证子线程已所有执行结束
	WaitForMultipleObjects(THREAD_NUM, handle, TRUE, INFINITE);  
	return 0;
}

unsigned int __stdcall Fun(void *pPM)
{
//因为创建线程是要一定的开销的,所以新线程并不能第一时间运行到这来
	int nThreadNum = *(int *)pPM; //子线程获取參数
	Sleep(50);//some work should to do
	g_nNum++;  //处理全局资源
	Sleep(0);//some work should to do
	printf("线程编号为%d  全局资源值为%d\n", nThreadNum, g_nNum);
	return 0;
}

执行结果能够參考下列图示,强烈建议读者亲自试一试。

1

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

2

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

3

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

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

 

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

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

秒杀多线程第七篇 经典线程同步 相互排斥量Mutex已经公布,欢迎參阅。

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

 

 

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

 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/118683.html原文链接:https://javaforall.net

(0)
上一篇 2021年12月2日 下午6:00
下一篇 2021年12月2日 下午7:00


相关推荐

  • 同步传输与异步传输相比_以下效率最高的数据交换控制方式

    同步传输与异步传输相比_以下效率最高的数据交换控制方式在网络通信过程中,通信双方要交换数据,需要高度的协同工作。为了正确的解释信号,接收方必须确切地知道信号应当何时接收和处理,因此定时是至关重要的。在计算机网络中,定时的因素称为位同步。同步是要接收方按照发送方发送的每个位的起止时刻和速率来接收数据,否则会产生误差。通常可以采用同步或异步的传输方式对位进行同步处理。1.异步传输(AsynchronousTransmission):异步传输将比

    2025年11月22日
    4
  • 查看服务时提示“一个或多个ActiveX控件无法显示”的解决方法

    查看服务时提示“一个或多个ActiveX控件无法显示”的解决方法最近,查看服务(services.msc)时,默认使用扩展视图时会提示一个或多个ActiveX控件无法显示,原因可能是下列其中之一:1)当前安全设置禁止运行此页面中的ActiveX控件,或2)

    2025年8月21日
    4
  • 闭包面试题原题_闭包 数学

    闭包面试题原题_闭包 数学闭包面试题原题functionfun(n,o){//①console.log(o);return{//②fun:function(m){//③returnfun(m,n);//④}};}//第一个例子vara=fun(0);/…

    2022年10月21日
    5
  • 电路原理图中的“NC“是什么意思?

    电路原理图中的“NC“是什么意思?电路原理图中的 NC 是什么意思 1 在看电路原理图的时候 电路原理图上有 NC 我查了下是表示 此处不贴任何电子器件 我看了下实际的电路板确实没贴 那么不贴的话是不是相当于这个位置空出来了 断路 了 就是说芯片的这个引脚在电路中是 悬空状态的 与其他任意点都没有连接 2 在模拟电路中 nc 是常闭的符号 NC 是英文 normallyclos 的缩写 意思是指在不通电时处于闭合导通状态的触点 通常称为常闭触点 常开的符号是 NO 英文名称为 normalopen 对应的英文字母 open 的首个字母 O

    2026年3月19日
    1
  • 分布式文件存储的实现

    分布式文件存储的实现分布式文件存储的实现

    2022年5月9日
    44
  • android 苹果 换机,苹果12怎么一键换机安卓?iPhone12一键换机功能操作步骤

    android 苹果 换机,苹果12怎么一键换机安卓?iPhone12一键换机功能操作步骤苹果12怎么从安卓一键换机?相信有很多朋友最近新购了iPhone12,但是原来安卓手机中有不少的联系人和文件资料,要是手动来备份还原,那就太麻烦了,所以这时候就需要用到一键换机功能了。下面我们就为大家带来了苹果12电脑端和手机端的一键换机教程,一起来看看吧!苹果12怎么一键换机安卓?iPhone12一键换机功能操作步骤苹果一键换机功能怎么操作1、使用QQ同步助手换机,我们可以在原来安卓或者旧苹果手…

    2022年5月26日
    119

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号