C#多线程学习之(六)互斥对象用法实例

本文实例讲述了C#多线程学习之互斥对象用法。分享给大家供大家参考。具体分析如下:

如何控制好多个线程相互之间的联系,不产生冲突和重复,这需要用到互斥对象,即:System.Threading 命名空间中的 Mutex 类。

我们可以把Mutex看作一个出租车,乘客看作线程。乘客首先等车,然后上车,最后下车。当一个乘客在 车上时,其他乘客就只有等他下车以后才可以上车。而线程与Mutex对象的关系也正是如此,线程使用Mutex.WaitOne()方法等待Mutex对 象被释放,如果它等待的Mutex对象被释放了,它就自动拥有这个对象,直到它调用Mutex.ReleaseMutex()方法释放这个对象,而在此期 间,其他想要获取这个Mutex对象的线程都只有等待。

下面这个例子使用了Mutex对象来同步四个线程,主线程等待四个线程的结束,而这四个线程的运行又是与两个Mutex对象相关联的。

其中还用到AutoResetEvent类的对象,可以把它理解为一个信号灯。这里用它的有信号状态来表示一个线程的结束。
AutoResetEvent.Set()方法设置它为有信号状态
AutoResetEvent.Reset()方法设置它为无信号状态

Mutex 类的程序示例:

using System;
using System.Threading;
namespace ThreadExample
{
 public class MutexSample
 {
 static Mutex gM1;
 static Mutex gM2;
 const int ITERS = 100;
 static AutoResetEvent Event1 = new AutoResetEvent(false);
 static AutoResetEvent Event2 = new AutoResetEvent(false);
 static AutoResetEvent Event3 = new AutoResetEvent(false);
 static AutoResetEvent Event4 = new AutoResetEvent(false);
 public static void Main(String[] args)
 {
 Console.WriteLine("Mutex Sample ");
 //创建一个Mutex对象,并且命名为MyMutex
 gM1 = new Mutex(true,"MyMutex");
 //创建一个未命名的Mutex 对象.
 gM2 = new Mutex(true);
 Console.WriteLine(" - Main Owns gM1 and gM2");
 AutoResetEvent[] evs = new AutoResetEvent[4];
 evs[0] = Event1; //为后面的线程t1,t2,t3,t4定义AutoResetEvent对象
 evs[1] = Event2;
 evs[2] = Event3;
 evs[3] = Event4;
 MutexSample tm = new MutexSample( );
 Thread t1 = new Thread(new ThreadStart(tm.t1Start));
 Thread t2 = new Thread(new ThreadStart(tm.t2Start));
 Thread t3 = new Thread(new ThreadStart(tm.t3Start));
 Thread t4 = new Thread(new ThreadStart(tm.t4Start));
 t1.Start( );
 //使用Mutex.WaitAll()方法等待一个Mutex数组中的对象全部被释放
 t2.Start( );
 //使用Mutex.WaitOne()方法等待gM1的释放
 t3.Start( );
 //使用Mutex.WaitAny()方法等待一个Mutex数组中任意一个对象被释放
 t4.Start( );
 //使用Mutex.WaitOne()方法等待gM2的释放
 Thread.Sleep(2000);
 Console.WriteLine(" - Main releases gM1");
 gM1.ReleaseMutex( );
 //线程t2,t3结束条件满足
 Thread.Sleep(1000);
 Console.WriteLine(" - Main releases gM2");
 gM2.ReleaseMutex( );
 //线程t1,t4结束条件满足
 //等待所有四个线程结束
 WaitHandle.WaitAll(evs);
 Console.WriteLine(" Mutex Sample");
 Console.ReadLine();
 }
 public void t1Start( )
 {
 Console.WriteLine("t1Start started, Mutex.WaitAll(Mutex[])");
 Mutex[] gMs = new Mutex[2];
 gMs[0] = gM1;
 //创建一个Mutex数组作为Mutex.WaitAll()方法的参数
 gMs[1] = gM2;
 Mutex.WaitAll(gMs);
 //等待gM1和gM2都被释放
 Thread.Sleep(2000);
 Console.WriteLine("t1Start finished,Mutex.WaitAll(Mutex[]) satisfied");
 Event1.Set( );
 //线程结束,将Event1设置为有信号状态
 }
 public void t2Start( )
 {
 Console.WriteLine("t2Start started, gM1.WaitOne( )");
 gM1.WaitOne( );//等待gM1的释放
 Console.WriteLine("t2Start finished, gM1.WaitOne( ) satisfied");
 Event2.Set( );//线程结束,将Event2设置为有信号状态
 }
 public void t3Start( )
 {
 Console.WriteLine("t3Start started, Mutex.WaitAny(Mutex[])");
 Mutex[] gMs = new Mutex[2];
 gMs[0] = gM1;//创建一个Mutex数组作为Mutex.WaitAny()方法的参数
 gMs[1] = gM2;
 Mutex.WaitAny(gMs);//等待数组中任意一个Mutex对象被释放
 Console.WriteLine("t3Start finished, Mutex.WaitAny(Mutex[])");
 Event3.Set( );//线程结束,将Event3设置为有信号状态
 }
 public void t4Start( )
 {
 Console.WriteLine("t4Start started, gM2.WaitOne( )");
 gM2.WaitOne( );//等待gM2被释放
 Console.WriteLine("t4Start finished, gM2.WaitOne( )");
 Event4.Set( );//线程结束,将Event4设置为有信号状态
 }
 }
}

程序的输出结果:

Mutex Sample
 - Main Owns gM1 and gM2
t1Start started, Mutex.WaitAll(Mutex[])
t2Start started, gM1.WaitOne( )
t3Start started, Mutex.WaitAny(Mutex[])
t4Start started, gM2.WaitOne( )
 - Main releases gM1
t2Start finished, gM1.WaitOne( ) satisfied
t3Start finished, Mutex.WaitAny(Mutex[])
 - Main releases gM2
t1Start finished, Mutex.WaitAll(Mutex[]) satisfied
t4Start finished, gM2.WaitOne( )
 Mutex Sample

从执行结果可以很清楚地看到,线程t2,t3的运行是以gM1的释放为条件的,而t4在gM2释放后开始 执行,t1则在gM1和gM2都被释放了之后才执行。Main()函数最后,使用WaitHandle等待所有的AutoResetEvent对象的信 号,这些对象的信号代表相应线程的结束。

希望本文所述对大家的C#程序设计有所帮助。

(0)

相关推荐

  • C#多线程编程之使用ReaderWriterLock类实现多用户读与单用户写同步的方法

    本文实例讲述了C#多线程编程之使用ReaderWriterLock类实现多用户读与单用户写同步的方法.分享给大家供大家参考,具体如下: 摘要:C#提供了System.Threading.ReaderWriterLock类以适应多用户读/单用户写的场景.该类可实现以下功能:如果资源未被写操作锁定,那么任何线程都可对该资源进行读操作锁定,并且对读操作锁数量没有限制,即多个线程可同时对该资源进行读操作锁定,以读取数据. 使用Monitor或Mutex进行同步控制的问题:由于独占访问模型不允许任何形式的

  • C#多线程编程中的锁系统基本用法

    平常在多线程开发中,总避免不了线程同步.本篇就对net多线程中的锁系统做个简单描述. 目录 一:lock.Monitor      1:基础.      2: 作用域.      3:字符串锁.      4:monitor使用 二:mutex 三:Semaphore 四:总结 一:lock.Monitor 1:基础 Lock是Monitor语法糖简化写法.Lock在IL会生成Monitor. 复制代码 代码如下: //======Example 1=====             strin

  • C#队列Queue多线程用法实例

    本文实例讲述了C#队列Queue多线程用法.分享给大家供大家参考.具体分析如下: 这里展示一个例子,供学习使用: private void button_测试Queue结合多线程_Click(object sender, EventArgs e) { Console.WriteLine("初始化队列"); queue = new Queue<string>(); string[] cars = new string[]{"宝马","奔驰&quo

  • C#在Unity游戏开发中进行多线程编程的方法

    在这之前,有很多人在质疑Unity支不支持多线程,事实上Unity是支持多线程的.而提到多线程就要提到Unity非常常用的协程,然而协程并非真正的多线程.协程其实是等某个操作完成之后再执行后面的代码,或者说是控制代码在特定的时机执行.而多线程在Unity渲染和复杂逻辑运算时可以高效的使用多核CPU,帮助程序可以更高效的运行.本篇主要介绍在Unity中如何使用多线程. 首先引入C#中使用多线程的类库 using System.Threading; 创建线程实例的四种方式 一.线程执行无参方法 构造

  • C#基于委托实现多线程之间操作的方法

    本文实例讲述了C#基于委托实现多线程之间操作的方法.分享给大家供大家参考,具体如下: 有的时候我们要起多个线程,更多的时候可能会有某个线程会去操作其他线程里的属性. 但是线程是并发的,一般的调用是无法实现我们的要求的. 于是,我们在这里就可以用委托,代码如下 private delegate void DelegateInfo(); private delegate void DelegateIsEnd(); //这个是线程调用其他线程的方法 private void Dowork() { //

  • C#实现向多线程传参的三种方式实例分析

    本文实例讲述了C#实现向多线程传参的三种方式.分享给大家供大家参考,具体如下: 从<C#高级编程>了解到给线程传递参数有两种方式,一种方式是使用带ParameterizedThreadStart委托参数的Thread构造函数,另一种方式是创建一个自定义类,把线程的方法定义为实例的方法,这样就可以初始化实例的数据,之后启动线程. 方式一:使用ParameterizedThreadStart委托 如果使用了ParameterizedThreadStart委托,线程的入口必须有一个object类型的

  • C#多线程编程中的锁系统(三)

    本章主要说下基于内核模式构造的线程同步方式,事件,信号量. 目录 一:理论 二:WaitHandle 三:AutoResetEvent 四:ManualResetEvent 五:总结 一:理论 我们晓得线程同步可分为,用户模式构造和内核模式构造. 内核模式构造:是由windows系统本身使用,内核对象进行调度协助的.内核对象是系统地址空间中的一个内存块,由系统创建维护. 内核对象为内核所拥有,而不为进程所拥有,所以不同进程可以访问同一个内核对象, 如进程,线程,作业,事件,文件,信号量,互斥量等

  • C#实现多线程写入同一个文件的方法

    本文实例讲述了C#实现多线程写入同一个文件的方法.分享给大家供大家参考.具体实现方法如下: namespace WfpApp { public partial class Form2 : Form { object obj = new object(); public Form2() { InitializeComponent(); System.Threading.Thread thread; string[] users = new string[] { "zkk", "

  • C#多线程编程中的锁系统(二)

    上章主要讲排他锁的直接使用方式.但实际当中全部都用锁又太浪费了,或者排他锁粒度太大了. 这一次我们说说升级锁和原子操作. 目录 1:volatile 2:  Interlocked 3:ReaderWriterLockSlim 4:总结 一:volatile 简单来说: volatile关键字是告诉c#编译器和JIT编译器,不对volatile标记的字段做任何的缓存.确保字段读写都是原子操作,最新值. 这不就是锁吗?   其这货它根本不是锁, 它的原子操作是基于CPU本身的,非阻塞的. 因为32

  • 解析C#多线程编程中异步多线程的实现及线程池的使用

    0.线程的本质 线程不是一个计算机硬件的功能,而是操作系统提供的一种逻辑功能,线程本质上是进程中一段并发运行的代码,所以线程需要操作系统投入CPU资源来运行和调度. 1.多线程: 使用多个处理句柄同时对多个任务进行控制处理的一种技术.据博主的理解,多线程就是该应用的主线程任命其他多个线程去协助它完成需要的功能,并且主线程和协助线程是完全独立进行的.不知道这样说好不好理解,后面慢慢在使用中会有更加详细的讲解. 2.多线程的使用: (1)最简单.最原始的使用方法:Thread oGetArgThre

随机推荐