多线程-互斥变量

第一个 CreateMutex

函数功能:创建互斥量(注意与事件Event的创建函数对比)

函数原型:

HANDLE  CreateMutex(

  LPSECURITY_ATTRIBUTESlpMutexAttributes,

  BOOLbInitialOwner,     

  LPCTSTRlpName

);

函数说明:

第一个参数表示安全控制,一般直接传入NULL。

第二个参数用来确定互斥量的初始拥有者。如果传入TRUE表示互斥量对象内部会记录创建它的线程的线程ID号并将递归计数设置为1,由于该线程ID非零,所以互斥量处于未触发状态。如果传入FALSE,那么互斥量对象内部的线程ID号将设置为NULL,递归计数设置为0,这意味互斥量不为任何线程占用,处于触发状态。

第三个参数用来设置互斥量的名称,在多个进程中的线程就是通过名称来确保它们访问的是同一个互斥量。

函数访问值:

成功返回一个表示互斥量的句柄,失败返回NULL。

第二个打开互斥量

函数原型:

HANDLE OpenMutex(

 DWORDdwDesiredAccess,

 BOOLbInheritHandle,

 LPCTSTRlpName     //名称

);

函数说明:

第一个参数表示访问权限,对互斥量一般传入MUTEX_ALL_ACCESS。详细解释可以查看MSDN文档。

第二个参数表示互斥量句柄继承性,一般传入TRUE即可。

第三个参数表示名称。某一个进程中的线程创建互斥量后,其它进程中的线程就可以通过这个函数来找到这个互斥量。

函数访问值:

成功返回一个表示互斥量的句柄,失败返回NULL。

第三个触发互斥量

函数原型:

BOOL  ReleaseMutex (HANDLEhMutex)

函数说明:

访问互斥资源前应该要调用等待函数,结束访问时就要调用ReleaseMutex()来表示自己已经结束访问,其它线程可以开始访问了。

最后一个清理互斥量

由于互斥量是内核对象,因此使用CloseHandle()就可以(这一点所有内核对象都一样)。

#include<stdio.h>
#include<process.h>
#include<windows.h>
HANDLE hMutex;
int tickets=100;

DWORD WINAPI Fun1(VOID *lp)
{
	while(true)
	{
        WaitForSingleObject(hMutex,INFINITE);
		if(tickets>0)
		{
			printf("thread1 %d\n",tickets--);
			ReleaseMutex(hMutex);
		}
		else
		{
			ReleaseMutex(hMutex);
			break;
		}
	}
	return 0;
}

DWORD WINAPI Fun2(VOID *lp)
{
	while(true)
	{
       WaitForSingleObject(hMutex,INFINITE);
	   if(tickets>0)
	   {
		   printf("thread2 %d\n",tickets--);
		   ReleaseMutex(hMutex);
	   }
	   else
	   {
		   ReleaseMutex(hMutex);
		   break;
	   }
	}
	return 0;
}

int main()
{
	hMutex=CreateMutex(NULL,FALSE,NULL);

	HANDLE handle1=CreateThread(NULL,0,Fun1,NULL,0,NULL);
	HANDLE handle2=CreateThread(NULL,0,Fun2,NULL,0,NULL);
	CloseHandle(handle1);
	CloseHandle(handle2);
    Sleep(3000);
	CloseHandle(hMutex);
	return 0;
}

参考:http://blog.csdn.net/morewindows/article/details/7470936

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏黑泽君的专栏

day19_java基础加强_动态代理+注解+类加载器

        Proxy Pattern(即:代理模式),23种常用的面向对象软件的设计模式之一。         代理模式的定义:为其他对象提供一种代理以控...

1174
来自专栏Java编程技术

ClassLoader解惑

一个Java程序要想运行起来,首先需要经过编译生成 .class文件,然后创建一个运行环境(jvm)来加载字节码文件到内存运行,而.class 文件是怎样被加载...

1191
来自专栏腾讯IVWEB团队的专栏

踩坑记:当 JavaScript 遇上 UINT 64

写下这篇文章的缘由是因为在项目过程中,碰到了一个使用 JavaScript 处理 UINT64 类型数字的坑。二进制浮点数中的 0.1 和 0.2 并不是十分...

7720
来自专栏博客园

X--名称空间详解

X名称空间里面的成员(如X:Name,X:Class)都是写给XAML编译器看的、用来引导XAML代码将XAML代码编译为CLR代码。

1682
来自专栏JavaEdge

Netty 源码深度解析(八) - 解码

就像很多标准的架构模式都被各种专用框架所支持一样,常见的数据处理模式往往也是目标实现的很好的候选对象,它可以节省开发人员大量的时间和精力。

1781
来自专栏鸿的学习笔记

Python写的Python解释器(七)--完结篇

在程序运行时,只会创建一次VirtualMachine实例,这是因为只有一个Python解释器。 VirtualMachine存储着call stack,异常状...

1033
来自专栏阮一峰的网络日志

ES6 的功能侦测库 ES-Checker

两周前,《ECMAScript国际标准(第6版)》正式通过,下一代 JavaScript 语言定案。 ? ECMAScript 6(以下简称 ES6)总共新增了...

3697
来自专栏君赏技术博客

Object-C中的黑魔法

在Swift中存在Option类型,也就是使用?和!声明的变量。但是OC里面没有这个特征,因为在XCODE6.3之后出现新的关键词定义用于OC转SWIFT时候可...

1791
来自专栏用户2442861的专栏

初学Redis(3)——简单实现Redis缓存中的排序功能

http://blog.csdn.net/qtyl1988/article/details/39545531

631
来自专栏岑玉海

Hbase 学习(三)Coprocessors

Coprocessors 之前我们的filter都是在客户端定义,然后传到服务端去执行的,这个Coprocessors是在服务端定义,在客户端调用,然后在服...

41411

扫码关注云+社区

领取腾讯云代金券