C#使用互斥量(Mutex)实现多进程并发操作时进程间的同步操作(进程同步)
本文主要是实现操作系统级别的多进程间线程同步(进程同步)的示例代码及测试结果。代码经过测试,可供参考,也可直接使用。
承接上一篇博客的业务场景[C#使用读写锁三行代码简单解决多线程并发写入文件时线程同步的问题]。
随着服务进程的增多,光凭进程内的线程同步已经不能满足现在的需求,导致多进程同时写入同一个文件时,一样提示文件被占用的问题。
在这种场景下,跨进程级的锁是不可避免的。在.NET提供的参考中,进程锁都继承了System.Threading.WaitHandle类。
而在本文中针对单个文件同一时间仅允许单个进程(线程)操作的场景,System.Threading.Mutex类无疑是最简单也是最合适的选择。
该类型的对象可以使用命名(字符串)互斥量实现当前会话级或操作系统级的同步需求。我选择了操作系统级别的同步编写示例,因为覆盖面更广。
下面是实现代码,注释很详细就不细说了:
namespace WaitHandleExample
{
class Program
{
static void Main(string[] args)
{
#region 简单使用
//var mutexKey = MutexExample.GetFilePathMutexKey("文件路径");
//MutexExample.MutexExec(mutexKey, () =>
//{
// Console.WriteLine("需要进程同步执行的代码");
//});
#endregion #region 测试代码
var filePath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "test.log").ToUpper();
var mutexKey = MutexExample.GetFilePathMutexKey(filePath); //同时开启N个写入线程
Parallel.For(, LogCount, e =>
{
//没使用互斥锁操作写入,大量写入错误;FileStream包含FileShare的构造函数也仅实现了进程内的线程同步,多进程同时写入时也会出错
//WriteLog(filePath); //使用互斥锁操作写入,由于同一时间仅有一个线程操作,所以不会出错
MutexExample.MutexExec(mutexKey, () =>
{
WriteLog(filePath);
});
}); Console.WriteLine(string.Format("Log Count:{0}.\t\tWrited Count:{1}.\tFailed Count:{2}.", LogCount.ToString(), WritedCount.ToString(), FailedCount.ToString()));
Console.Read();
#endregion
} /// <summary>
/// C#互斥量使用示例代码
/// </summary>
/// <remarks>已在经过测试并上线运行,可直接使用</remarks>
public static class MutexExample
{
/// <summary>
/// 进程间同步执行的简单例子
/// </summary>
/// <param name="action">同步处理代码</param>
/// <param name="mutexKey">操作系统级的同步键
/// (如果将 name 指定为 null 或空字符串,则创建一个局部互斥体。
/// 如果名称以前缀“Global\”开头,则 mutex 在所有终端服务器会话中均为可见。
/// 如果名称以前缀“Local\”开头,则 mutex 仅在创建它的终端服务器会话中可见。
/// 如果创建已命名 mutex 时不指定前缀,则它将采用前缀“Local\”。)</param>
/// <remarks>不重试且不考虑异常情况处理的简单例子</remarks>
[Obsolete(error: false, message: "请使用MutexExec")]
public static void MutexExecEasy(string mutexKey, Action action)
{
//声明一个已命名的互斥体,实现进程间同步;该命名互斥体不存在则自动创建,已存在则直接获取
using (Mutex mut = new Mutex(false, mutexKey))
{
try
{
//上锁,其他线程需等待释放锁之后才能执行处理;若其他线程已经上锁或优先上锁,则先等待其他线程执行完毕
mut.WaitOne();
//执行处理代码(在调用WaitHandle.WaitOne至WaitHandle.ReleaseMutex的时间段里,只有一个线程处理,其他线程都得等待释放锁后才能执行该代码段)
action();
}
finally
{
//释放锁,让其他进程(或线程)得以继续执行
mut.ReleaseMutex();
}
}
} /// <summary>
/// 获取文件名对应的进程同步键
/// </summary>
/// <param name="filePath">文件路径(请注意大小写及空格)</param>
/// <returns>进程同步键(互斥体名称)</returns>
public static string GetFilePathMutexKey(string filePath)
{
//生成文件对应的同步键,可自定义格式(互斥体名称对特殊字符支持不友好,遂转换为BASE64格式字符串)
var fileKey = Convert.ToBase64String(Encoding.Default.GetBytes(string.Format(@"FILE\{0}", filePath)));
//转换为操作系统级的同步键
var mutexKey = string.Format(@"Global\{0}", fileKey);
return mutexKey;
} /// <summary>
/// 进程间同步执行
/// </summary>
/// <param name="mutexKey">操作系统级的同步键
/// (如果将 name 指定为 null 或空字符串,则创建一个局部互斥体。
/// 如果名称以前缀“Global\”开头,则 mutex 在所有终端服务器会话中均为可见。
/// 如果名称以前缀“Local\”开头,则 mutex 仅在创建它的终端服务器会话中可见。
/// 如果创建已命名 mutex 时不指定前缀,则它将采用前缀“Local\”。)</param>
/// <param name="action">同步处理操作</param>
public static void MutexExec(string mutexKey, Action action)
{
MutexExec(mutexKey: mutexKey, action: action, recursive: false);
} /// <summary>
/// 进程间同步执行
/// </summary>
/// <param name="mutexKey">操作系统级的同步键
/// (如果将 name 指定为 null 或空字符串,则创建一个局部互斥体。
/// 如果名称以前缀“Global\”开头,则 mutex 在所有终端服务器会话中均为可见。
/// 如果名称以前缀“Local\”开头,则 mutex 仅在创建它的终端服务器会话中可见。
/// 如果创建已命名 mutex 时不指定前缀,则它将采用前缀“Local\”。)</param>
/// <param name="action">同步处理操作</param>
/// <param name="recursive">指示当前调用是否为递归处理,递归处理时检测到异常则抛出异常,避免进入无限递归</param>
private static void MutexExec(string mutexKey, Action action, bool recursive)
{
//声明一个已命名的互斥体,实现进程间同步;该命名互斥体不存在则自动创建,已存在则直接获取
//initiallyOwned: false:默认当前线程并不拥有已存在互斥体的所属权,即默认本线程并非为首次创建该命名互斥体的线程
//注意:并发声明同名的命名互斥体时,若间隔时间过短,则可能同时声明了多个名称相同的互斥体,并且同名的多个互斥体之间并不同步,高并发用户请另行处理
using (Mutex mut = new Mutex(initiallyOwned: false, name: mutexKey))
{
try
{
//上锁,其他线程需等待释放锁之后才能执行处理;若其他线程已经上锁或优先上锁,则先等待其他线程执行完毕
mut.WaitOne();
//执行处理代码(在调用WaitHandle.WaitOne至WaitHandle.ReleaseMutex的时间段里,只有一个线程处理,其他线程都得等待释放锁后才能执行该代码段)
action();
}
//当其他进程已上锁且没有正常释放互斥锁时(譬如进程忽然关闭或退出),则会抛出AbandonedMutexException异常
catch (AbandonedMutexException ex)
{
//避免进入无限递归
if (recursive)
throw ex; //非递归调用,由其他进程抛出互斥锁解锁异常时,重试执行
MutexExec(mutexKey: mutexKey, action: action, recursive: true);
}
finally
{
//释放锁,让其他进程(或线程)得以继续执行
mut.ReleaseMutex();
}
}
}
} #region 测试写文件的代码
static int LogCount = ;
static int WritedCount = ;
static int FailedCount = ;
static void WriteLog(string logFilePath)
{
try
{
var now = DateTime.Now;
var logContent = string.Format("Tid: {0}{1} {2}.{3}\r\n", Thread.CurrentThread.ManagedThreadId.ToString().PadRight(), now.ToLongDateString(), now.ToLongTimeString(), now.Millisecond.ToString());
File.AppendAllText(logFilePath, logContent);
WritedCount++;
}
catch (Exception ex)
{
Console.WriteLine(ex.Message);
FailedCount++;
}
}
#endregion
}
}
测试不使用进程同步,多进程多线程同时写入文件:
测试结果:6个进程同时进行3000次写入请求,仅成功写入277次
测试使用互斥量进行进程同步,多进程多线程同时写入文件:
测试结果:6个进程同时进行3000次写入请求,全部成功写入
补充:
进程同步的资源消耗及效率比线程同步要差得多,请根据实际场景合理使用。
本文虽然是用写入文件作为示例,但进程同步的代码使用场景与文件操作无关。
Semaphore类(信号灯)虽然可以限制同时操作的线程数,甚至把最大同时操作数设置为1时,行为与Mutex类(互斥量)类似;但是由于信号灯在其他进程中出现异常退出时并不能接收到异常通知,只能通过等待超时触发异常,并不适合现在的场景,所以并没讲述。
关于进程同步的其他深入了解及应用,请参阅其他资料。
C#使用互斥量(Mutex)实现多进程并发操作时进程间的同步操作(进程同步)的更多相关文章
- C#使用互斥量(Mutex)实现多进程并发操作时多进程间线程同步操作(进程同步)的简单示例代码及使用方法
本文主要是实现操作系统级别的多进程间线程同步(进程同步)的示例代码及测试结果.代码经过测试,可供参考,也可直接使用. 承接上一篇博客的业务场景[C#使用读写锁三行代码简单解决多线程并发写入文件时线程同 ...
- 多线程相关------互斥量Mutex
互斥量(Mutex) 互斥量是一个可以处于两态之一的变量:解锁和加锁.只有拥有互斥对象的线程才具有访问资源的权限.并且互斥量可以用于不同进程中的线程的互斥访问. 相关函数: CreateMutex用于 ...
- 经典线程同步 互斥量Mutex
阅读本篇之前推荐阅读以下姊妹篇: <秒杀多线程第四篇一个经典的多线程同步问题> <秒杀多线程第五篇经典线程同步关键段CS> <秒杀多线程第六篇经典线程同步事件Event& ...
- [一个经典的多线程同步问题]解决方案三:互斥量Mutex
本篇通过互斥量来解决线程的同步,学习其中的一些知识. 互斥量也是一个内核对象,它用来确保一个线程独占一个资源的访问.互斥量与关键段的行为非常相似,并且互斥量可以用于不同进程中的线程互斥访问资源.使用互 ...
- (转)经典线程同步 互斥量Mutex
阅读本篇之前推荐阅读以下姊妹篇: <秒杀多线程第四篇一个经典的多线程同步问题> <秒杀多线程第五篇经典线程同步关键段CS> <秒杀多线程第六篇经典线程同步事件Event& ...
- 多线程面试题系列(7):经典线程同步 互斥量Mutex
前面介绍了关键段CS.事件Event在经典线程同步问题中的使用.本篇介绍用互斥量Mutex来解决这个问题. 互斥量也是一个内核对象,它用来确保一个线程独占一个资源的访问.互斥量与关键段的行为非常相似, ...
- C#线程同步(3)- 互斥量 Mutex
文章原始出处 http://xxinside.blogbus.com/logs/47162540.html 预备知识:C#线程同步(1)- 临界区&Lock,C#线程同步(2)- 临界区&am ...
- 秒杀多线程第七篇 经典线程同步 互斥量Mutex
本文转载于:http://blog.csdn.net/morewindows/article/details/7470936 前面介绍了关键段CS.事件Event在经典线程同步问题中的使用.本篇介绍用 ...
- windows多线程(六) 互斥量Mutex与关键段CriticalSection比较
一.关键段CS 和 互斥量Mutex 的相同点:都有线程拥有权 关键段和互斥量都有线程拥有权,即可以被一个线程拥有.在 前面讲关键段CS的文章中有说到,关键段结构体的第四个参数保存着拥有该关键段的线程 ...
随机推荐
- 201521123113 《Java程序设计》第1周学习总结
1. 本章学习总结 1.java是一个面向对象的编程语言,相对于c++来说代码较简便又好用.第一次接触java时感觉每句代码比较难写,但学习了一些快捷方法后就很方便了. 2.java运行于JVM,因此 ...
- 201521123059 《Java程序设计》第十二周学习总结
1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结多流与文件相关内容. 2. 书面作业 将Student对象(属性:int id, String name,int age,doubl ...
- 201521123009 《Java程序设计》第10周学习总结
1. 本周学习总结 2. 书面作业 本次PTA作业题集异常.多线程 Q1:finally 题目4-2 1.1 截图你的提交结果(出现学号) 1.2 4-2中finally中捕获异常需要注意什么? tr ...
- 在linux环境下搭建java web测试环境(非常详细!!)
一.项目必备软件及基本思路 项目必备:虚拟机:VMware Workstation (已安装linux的 CentOS6.5版本) 项目:java web项目 (必须在本地部署编译后选择项目的webR ...
- 【京东账户】——Mysql/PHP/Ajax爬坑之页头页尾加载
一.引言 实现京东的账户项目,有一个小功能,页头页尾加载.要用到的是Apach环境,Mysql.PHP以及Ajax. 二.实现 原理: 用php文件分别写一个的页头和一个页尾,放在前后两个div里. ...
- 指定路径下建立Access数据库并插入数据
今天刚刚开通博客,想要把我这几天完成小任务的过程,记录下来.我从事软件开发的时间不到1年,写的不足之处,还请前辈们多多指教. 上周四也就是2016-04-14号上午,部门领导交给我一个小任务,概括来讲 ...
- 记录各种IE兼容问题,IE6,IE7,IE8,IE9,IE10
记录遇到的IE BUG: 1.IE8开发者工具打不开 解决办法:IE8新增了开发人员工具,非常不错,比早期的DevToolbar好用多了.不过在我的Win7下 使用的时候偶尔会出现一个莫名其妙的问 ...
- JQuery&原生js ——实现剪刀石头布小游戏
前言 jQuery是一个快速.简洁的JavaScript框架,是继Prototype之后又一个优秀的JavaScript代码库( 或JavaScript框架).jQuery设计的宗旨是“write L ...
- pongo英雄会-幸运数题解
显然我们只要知道1~x范围有多少幸运数(用f(x)表示),lucky(x,y)=f(y)-f(x-1). 解法1. 计算排列数 由于y<=1000000000这个规模,我们不能暴力验证每个数是否 ...
- C# 7 局部函数剖析
局部函数是C# 7中的一个新功能,允许在一个函数中定义另一个函数. 何时使用局部函数? 局部函数的主要功能与匿名方法非常相似:在某些情况下,创建一个命名函数在读者的认知负担方面代价太大.有时,函数本身 ...