C#线程学习笔记六:线程同步--信号量和互斥体
本笔记摘抄自:https://www.cnblogs.com/zhili/archive/2012/07/23/Mutex_And_Semaphore.html,记录一下学习过程以备后续查用。
一、信号量(Semaphore)
信号量(Semaphore)是由内核对象维护的int变量。当信号量为0时,在信号量上等待的线程会堵塞;信号量大于0时,就解除堵塞。当在一个信号量上等待
的线程解除堵塞时,内核自动会将信号量的计数减1。在.NET下通过Semaphore类来实现信号量同步。
Semaphore类限制可同时访问某一资源或资源池的线程数。线程通过调用 WaitOne方法将信号量减1,并通过调用Release方法把信号量加1。
先说下构造函数:
public Semaphore(int initialCount,int maximumCount);通过两个参数来设置信号的初始计数和最大计数。
下面代码演示信号量同步的使用:
class Program
{
//共享资源
public static int number = ;
//初始信号量计数为0,最大计数为10。
public static Semaphore semaphore = new Semaphore(, );
static void Main(string[] args)
{
#region 线程同步:使用信号量实现同步
for (int i = ; i < ; i++)
{
Thread thread = new Thread(new ParameterizedThreadStart(SemaphoreMethod));
thread.Start(i);
} //每次增加2个信号量,即每次释放2个线程。
for (int j = ; j < ; j++)
{
Console.WriteLine("红灯转绿灯……");
semaphore.Release();
Thread.Sleep();
}
Console.Read();
#endregion
} /// <summary>
/// Semaphore方法
/// </summary>
public static void SemaphoreMethod(object parameter)
{
while ((int)parameter != number)
{
Thread.Sleep();
}
//信号量计数减1
semaphore.WaitOne();
Console.WriteLine("The current value of number is:{0}", ++number);
}
}
运行结果如下:

与上一篇AutoResetEvent类似,信号量也可以实现跨进程间的线程同步。通过调用public Semaphore(int initialCount,int maximumCount,string name);
构造函数,传入一个信号量名来实现此功能。
下面代码演示跨进程间的线程同步:
第一个进程代码:
class Program
{
//共享资源
public static int number = ; //初始信号量计数为0,最大计数为10。
public static Semaphore semaphore1 = new Semaphore(, , "Semaphore1");
public static Semaphore semaphore2 = new Semaphore(, , "Semaphore2"); static void Main(string[] args)
{
#region 线程同步:使用信号量实现跨进程之间的线程同步
for (int i = ; i < ; i++)
{
Thread thread = new Thread(new ParameterizedThreadStart(Semaphore1Method));
thread.Start(i);
} //为了有时间去启动另外一个进程
Thread.Sleep(); //每次增加2个信号量,即每次释放2个线程。
for (int j = ; j < ; j++)
{
Console.WriteLine("信号灯1红灯转绿灯……");
semaphore1.Release();
Console.WriteLine("信号灯2红灯转绿灯……");
semaphore2.Release();
Thread.Sleep();
}
Console.Read();
#endregion
} /// <summary>
/// Semaphore1方法
/// </summary>
public static void Semaphore1Method(object parameter)
{
while ((int)parameter != number)
{
Thread.Sleep();
}
//信号量计数减1
semaphore1.WaitOne();
Console.WriteLine("Semaphore1:The current value of number is:{0}", ++number);
}
}
第二个进程代码:
class Program
{
//共享资源
public static int number = ;
//创建对象
public static Semaphore semaphore2 = new Semaphore(, , "Semaphore2"); static void Main(string[] args)
{
#region 通过信号量实现跨进程间的线程同步
for (int i = ; i < ; i++)
{
Thread thread = new Thread(new ParameterizedThreadStart(Semaphore2Method));
thread.Start(i);
}
Console.Read();
#endregion
} /// <summary>
/// Semaphore2方法
/// </summary>
public static void Semaphore2Method(object parameter)
{
while ((int)parameter != number)
{
Thread.Sleep();
}
//信号量计数减1
semaphore2.WaitOne();
Console.WriteLine("Semaphore2:The current value of number is:{0}", ++number);
}
}
运行结果如下:

从结果可以看出,第一个进程的semaphore2.Release(2);信号发出后,第二个进程可以收到并释放线程。
二、互斥体(Mutex)
Mutex对象是一个同步基元,当某一个线程占用Mutex对象时,其他也需要占用Mutex的线程将处于挂起状态。
下面代码演示互斥体同步的使用:
class Program
{
//共享资源
public static int number = ;
//互斥体
public static Mutex mutex = new Mutex(); static void Main(string[] args)
{
#region 线程同步:使用互斥体实现同步
for (int i = ; i < ; i++)
{
Thread thread = new Thread(MutexMethod);
thread.Start();
} Console.Read();
#endregion
} /// <summary>
/// Mutex方法
/// </summary>
public static void MutexMethod(object parameter)
{
mutex.WaitOne();
Thread.Sleep();
Console.WriteLine("The current value of number is:{0}", ++number);
mutex.ReleaseMutex();
}
}
运行结果如下:

下面代码演示跨进程间的线程同步:
第一个进程代码:
class Program
{
//共享资源
public static int number = ; //互斥体
public static Mutex mutex1 = new Mutex(false, "Mutex1");
public static Mutex mutex2 = new Mutex(false, "Mutex2"); static void Main(string[] args)
{
#region 线程同步:使用互斥体实现跨进程之间的线程同步
mutex1.WaitOne();
mutex2.WaitOne();
for (int i = ; i < ; i++)
{
Thread thread = new Thread(new ParameterizedThreadStart(Mutex1Method));
thread.Start(i);
} //为了有时间去启动另外一个进程
Thread.Sleep();
mutex1.ReleaseMutex();
mutex2.ReleaseMutex();
Console.Read();
#endregion
} /// <summary>
/// Mutex1方法
/// </summary>
public static void Mutex1Method(object parameter)
{
mutex1.WaitOne();
Thread.Sleep();
Console.WriteLine("Mutex1:The current value of number is:{0}", ++number);
mutex1.ReleaseMutex();
}
}
第二个进程代码:
class Program
{
//共享资源
public static int number = ;
//创建对象
public static Mutex mutex2 = new Mutex(false, "Mutex2"); static void Main(string[] args)
{
#region 通过互斥体实现跨进程之间的线程同步
for (int i = ; i < ; i++)
{
Thread thread = new Thread(new ParameterizedThreadStart(Mutex2Method));
thread.Start(i);
}
Console.Read();
#endregion
} /// <summary>
/// Mutex2方法
/// </summary>
public static void Mutex2Method(object parameter)
{
mutex2.WaitOne();
Thread.Sleep();
Console.WriteLine("Mutex2:The current value of number is:{0}", ++number);
mutex2.ReleaseMutex();
}
}
运行结果如下:

从结果可以看出,第一个进程的mutex2.ReleaseMutex();信号发出后,第二个进程可以收到并释放线程。
C#线程学习笔记六:线程同步--信号量和互斥体的更多相关文章
- JUC源码学习笔记5——线程池,FutureTask,Executor框架源码解析
JUC源码学习笔记5--线程池,FutureTask,Executor框架源码解析 源码基于JDK8 参考了美团技术博客 https://tech.meituan.com/2020/04/02/jav ...
- 操作系统学习笔记----进程/线程模型----Coursera课程笔记
操作系统学习笔记----进程/线程模型----Coursera课程笔记 进程/线程模型 0. 概述 0.1 进程模型 多道程序设计 进程的概念.进程控制块 进程状态及转换.进程队列 进程控制----进 ...
- C#线程学习笔记九:async & await入门二
一.异步方法返回类型 只能返回3种类型(void.Task和Task<T>). 1.1.void返回类型:调用方法执行异步方法,但又不需要做进一步的交互. class Program { ...
- java学习笔记15--多线程编程基础2
本文地址:http://www.cnblogs.com/archimedes/p/java-study-note15.html,转载请注明源地址. 线程的生命周期 1.线程的生命周期 线程从产生到消亡 ...
- Linux进程线程学习笔记:运行新程序
Linux进程线程学习笔记:运行新程序 周银辉 在上一篇中我们说到,当启动一个新进程以后,新进程会复制父进程的大部份上下 ...
- java之jvm学习笔记六-十二(实践写自己的安全管理器)(jar包的代码认证和签名) (实践对jar包的代码签名) (策略文件)(策略和保护域) (访问控制器) (访问控制器的栈校验机制) (jvm基本结构)
java之jvm学习笔记六(实践写自己的安全管理器) 安全管理器SecurityManager里设计的内容实在是非常的庞大,它的核心方法就是checkPerssiom这个方法里又调用 AccessCo ...
- Java IO学习笔记六:NIO到多路复用
作者:Grey 原文地址:Java IO学习笔记六:NIO到多路复用 虽然NIO性能上比BIO要好,参考:Java IO学习笔记五:BIO到NIO 但是NIO也有问题,NIO服务端的示例代码中往往会包 ...
- Learning ROS for Robotics Programming Second Edition学习笔记(六) indigo xtion pro live
中文译著已经出版,详情请参考:http://blog.csdn.net/ZhangRelay/article/category/6506865 Learning ROS for Robotics Pr ...
- Typescript 学习笔记六:接口
中文网:https://www.tslang.cn/ 官网:http://www.typescriptlang.org/ 目录: Typescript 学习笔记一:介绍.安装.编译 Typescrip ...
随机推荐
- python logger日志通用配置文件
阅读须知⚠️ 1.示例代码可直接放在项目py文件中即可使用 2.project_name,logfile_name变量需根据你的项目进行修改 3.日志输出格式format选择(可根据你的需要替换或修改 ...
- 洛谷P2670-扫雷游戏
文章目录 原题链接 题面简述 输入格式 输出格式 思路 代码 原题链接 题面简述 在n行m列的雷区中有一些格子含有地雷(称之为地雷格),其他格子不含地雷(称之为非地雷格).玩家翻开一个非地雷格时,该格 ...
- mr的partition分区
1.Partitioner 组件通过让 Map 对 Key 进行分区,从而将不同分区的 Key 交由不同的 Reduce 处理.Partition属于map端 2.分区的总数与任务的reduce任务数 ...
- CentOS 7 安装 bind 服务 实现内网DNS
目录 安装 配置 服务管理 测试 安装 废话不多说,直接安装 yum install -y bind bind-utils 配置 [root@jenkins named]# rpm -ql bind ...
- 《手把手教你》系列练习篇之7-python+ selenium自动化测试 -压轴篇(详细教程)
1. 简介 “压轴”原本是戏曲名词,指一场折子戏演出的倒数第二个剧目.在现代社会中有很多应用,比如“压轴戏”,但压轴也是人们知识的一个盲区.“压轴”本意是指倒数第二个节目,而不是人们常说的倒数第一个, ...
- springboot+apache前后端分离部署https
目录 1. 引言 2. 了解https.证书.openssl及keytool 2.1 https 2.1.1 什么是https 2.1.2 https解决什么问题 2.2 证书 2.2.1 证书内容 ...
- 转:Java logger组件:slf4j, jcl, jul, log4j, logback, log4j2
先说结论 建议优先使用logback 或 log4j2.log4j2 不建议和 slf4j 配合使用,因为格式转换会浪费性能. 名词:jcl 和 jul 标题中的 jcl 是 apache Jakar ...
- Android 通知的基本用法
一.概念 通知(Notification)是 Android 系统中比较有特色的一个功能,当某个应用程序希望像用户发送一些提示消息的时候,然而此时应用程序并不在前台运行,此时就可以借助通知来实现.发出 ...
- 模板引擎Thymeleaf
1.Thymeleaf简介 Thymeleaf 是一个跟 Velocity.FreeMarker 类似的模板引擎,它可以完全替代 JSP .相较与其他的模板引擎,它有如下三个极吸引人的特点 Thyme ...
- openlayers6结合geoserver实现地图空间查询(附源码下载)
前言 之前写过一篇 openlayers4 版本的地图空间查询文章,但是由于是封装一层 js 代码写的,很多初学者看起来比较有点吃力,所以本篇文章重新写一篇地图空间查询文章,直接基于最新版本 open ...