线程面试题之二:子线程之间的同步

本文探讨了C++中子线程之间的同步问题,通过使用CRITICAL_SECTION实现互斥访问关键代码段。文章介绍了初始化、销毁、进入和离开关键区域的四个函数,并强调了正确使用的重要性。
摘要由CSDN通过智能技术生成
创建两个子线程,子线程1循环 10 次,接着子线程2循环 10 次,接着又回到子线程1循环 10 次,接着再回到2又循环 10 次,如此循环50次,试写出代码。

(注:在写本篇学习笔记时,本人学习和参考了网络文章,并做了借鉴,感谢各位前辈的分享。如果本文对您有所帮助,您可以随意分享,如果发现文中有误,也请指教,谢谢。本文用到的调试工具:Microsoft Visual Studio 10,操作系统:Windows 7)

代码实现为:

#include "stdafx.h"
#include "process.h"
#include "windows.h"

static int gCount=0;
const int gSum=5;//总循环次数
const int gSleepTimes=100;
const int gEvyCount=3;

HANDLE g_hEventId[2];// 用于子线程之间的同步
HANDLE g_hFthEventId;// 用于父子线程的同步
CRITICAL_SECTION g_csThreadCode;
unsigned int __stdcall ThreadFunc(LPVOID pM)
{
	int i=0;
	int * w=(int *)pM;
	HANDLE h1;
	HANDLE h2;

	if (*w==0)
	{
		h1=g_hEventId[0];
		h2=g_hEventId[1];
	}
	else
	{
		h1=g_hEventId[1];
		h2=g_hEventId[0];
	}
	
	printf("子线程[%d]启动\n",GetCurrentThreadId());
	
	SetEvent(g_hFthEventId);
	while(TRUE)
	{
		WaitForSingleObject(h1, INFINITE);
		EnterCriticalSection(&g_csThreadCode);
		if (h1==g_hEventId[0])
		{
			gCount++;
		}
		if (gCount>gSum)
		{
			LeaveCriticalSection(&g_csThreadCode);
			SetEvent(h2);
			break;
		}
		for (i=0;i<gEvyCount;i++)
		{
			Sleep(gSleepTimes);
			printf("子线程[%d]第[%d]次循环[%d]次\n",GetCurrentThreadId(),gCount,i+1);
		}
		LeaveCriticalSection(&g_csThreadCode);
		SetEvent(h2);
	}
	printf("子线程[%d]接收到消息,任务完成退出\n",GetCurrentThreadId());

	return 0;
}

int _tmain(int argc, _TCHAR* argv[])
{
	printf("---------------------------------------------------------------------\n");
	printf("  子线程之间同步问题 子线程1每次循环%d次 子线程2每次循环%d次 总循环%d次\n",gEvyCount,gEvyCount,gSum);
	printf("---------------------------------------------------------------------\n");

	int i=0;
	HANDLE handleThreadId[2];
	
	InitializeCriticalSection(&g_csThreadCode);
	g_hFthEventId = CreateEvent(NULL, FALSE, FALSE, NULL);
	g_hEventId[0] = CreateEvent(NULL, FALSE, TRUE, NULL);
	g_hEventId[1] = CreateEvent(NULL, FALSE, FALSE, NULL);

	for(i=0;i<2;i++)
	{
		handleThreadId[i] = (HANDLE)_beginthreadex(NULL,0,ThreadFunc,&i,0,NULL);
		WaitForSingleObject(g_hFthEventId, INFINITE);
	}
	WaitForMultipleObjects(2,handleThreadId, TRUE, INFINITE);
	for(i=0;i<2;i++)
	{
		CloseHandle(handleThreadId[i]);
	}

	DeleteCriticalSection(&g_csThreadCode);
	CloseHandle(g_hFthEventId);
	CloseHandle(g_hEventId[0]);
	CloseHandle(g_hEventId[1]);
	printf("-----------OVER-------------\n");

	getchar();

	return 0;
}
运行结果为:


这道题目其实是对面试题一的扩展,由父子线程的同步变为子线程之间的同步。

在线程函数中,我们使用了关键代码段CriticalSection,使子线程之间对关键代码的操作互斥。

下面我们就来看下关键代码段的函数介绍:(以下介绍为转载,原文请参考:
http://blog.csdn.net/morewindows/article/details/7442639)

关键段CRITICAL_SECTION一共就四个函数,使用很是方便。下面是这四个函数的原型和使用说明。

函数功能:初始化

函数原型:

void InitializeCriticalSection(LPCRITICAL_SECTION lpCriticalSection);

函数说明:定义关键段变量后必须先初始化。

 

函数功能:销毁

函数原型:

void DeleteCriticalSection(LPCRITICAL_SECTION lpCriticalSection);

函数说明:用完之后记得销毁。

 

函数功能:进入关键区域

函数原型:

void EnterCriticalSection(LPCRITICAL_SECTION lpCriticalSection);

函数说明:系统保证各线程互斥的进入关键区域。

 

函数功能:离开关关键区域

函数原型:

void LeaveCriticalSection(LPCRITICAL_SECTION lpCriticalSection);


(注:在写本篇学习笔记时,本人学习和参考了网络文章,并做了借鉴,感谢各位前辈的分享。如果本文对您有所帮助,您可以随意分享,如果发现文中有误,也请指教,谢谢。本文用到的调试工具:Microsoft Visual Studio 10,操作系统:Windows 7)
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值