深入了解c# 信号量和互斥体

作者:Learning hard 时间:2022-09-19 11:07:47 

一、信号量(Semaphore)

信号量(Semaphore)是由内核对象维护的int变量,当信号量为0时,在信号量上等待的线程会堵塞,信号量大于0时,就解除堵塞。当在一个信号量上等待的线程解除堵塞时,内核自动会将信号量的计数减1。在.net 下通过Semaphore类来实现信号量同步。

Semaphore类限制可同时访问某一资源或资源池的线程数。WaitOne method, which is inherited from the WaitHandle class, and release the semaphore by calling the Release method.">线程通过调用 WaitOne方法将信号量减1,并通过调用 Release方法把信号量加1。

先说下构造函数:

public Semaphore(int initialCount,int maximumCount);通过两个参数来设置信号的初始计数和最大计数。

下面通过一段代码来演示信号量同步的使用:


class Program
 {
   // 初始信号量计数为0,最大计数为10
   public static Semaphore semaphore =new Semaphore(0,10);
   public static int time = 0;
   static void Main(string[] args)
   {
     for (int i = 0; i < 5; i++)
     {
       Thread test = new Thread(new ParameterizedThreadStart(TestMethod));

// 开始线程,并传递参数
       test.Start(i);
     }

// 等待1秒让所有线程开始并阻塞在信号量上
     Thread.Sleep(500);

// 信号量计数加4
     // 最后可以看到输出结果次数为4次
     semaphore.Release(4);
     Console.Read();    
   }

public static void TestMethod(object number)
   {
     // 设置一个时间间隔让输出有顺序
     int span = Interlocked.Add(ref time, 100);
     Thread.Sleep(1000 + span);

//信号量计数减1
     semaphore.WaitOne();

Console.WriteLine("Thread {0} run ", number);
   }
 }

运行结果:

深入了解c# 信号量和互斥体

同样信号量也可以实现进程中线程的同步,同样也是通过对信号量命名来实现的,

通过调用public Semaphore(int initialCount,int maximumCount,string name);该构造函数多传入一个信号量名来实现

下面一段实例代码来演示下:


using System;
using System.Threading;

namespace SemaphoreSample
{
 class Program
 {
   // 初始信号量计数为4,最大计数为10
   public static Semaphore semaphore =new Semaphore(4,10,"My");
   public static int time = 0;
   static void Main(string[] args)
   {
     for (int i = 0; i < 3; i++)
     {
       Thread test = new Thread(new ParameterizedThreadStart(TestMethod));

// 开始线程,并传递参数
       test.Start(i);
     }

// 等待1秒让所有线程开始并阻塞在信号量上
     Thread.Sleep(1000);

Console.Read();    
   }

public static void TestMethod(object number)
   {
     // 设置一个时间间隔让输出有顺序
     int span = Interlocked.Add(ref time, 500);
     Thread.Sleep(1000 + span);

//信号量计数减1
     semaphore.WaitOne();

Console.WriteLine("Thread {0} run ", number);
   }
 }
}

运行结果:

深入了解c# 信号量和互斥体

从运行结果中可以看出, 第二个进程值运行了一行语句, 因为我们设置的初始信号计数为4,每运行一个线程,信号计数通过调用WaitOne方法减1,所以第二个进行一开始信号计数为1而不是进程一中的4,如果我们把信号计数后面的name参数去除的话,此时第二个进程和第一个进程中的结果应该是一样的(因为此时没有进行不同进程中线程的同步)。

二、互斥体(Mutex)

同样互斥体也是同样可以实现线程之间的同步和不同进程中线程的同步的

先看看线程之间的同步的例子吧(在这里我也不多做解释了,因为他们之间的使用很类似,直接贴出代码):


class Program
 {
   public static Mutex mutex = new Mutex();
   public static int count;

static void Main(string[] args)
   {
     for (int i = 0; i < 10; i++)
     {
       Thread test = new Thread(TestMethod);

// 开始线程,并传递参数
       test.Start();
     }

Console.Read();
   }

public static void TestMethod()
   {
     mutex.WaitOne();
     Thread.Sleep(500);
     count++;
     Console.WriteLine("Current Cout Number is {0}", count);
     mutex.ReleaseMutex();
   }
 }

运行结果:

深入了解c# 信号量和互斥体

实现进程间同步:


class Program
 {
   public static Mutex mutex = new Mutex(false,"My");

static void Main(string[] args)
   {
     Thread t = new Thread(TestMethod);
     t.Start();

Console.Read();
   }

public static void TestMethod()
   {
     mutex.WaitOne();
     Thread.Sleep(5000);
     Console.WriteLine("Method start at : " + DateTime.Now.ToLongTimeString());
     mutex.ReleaseMutex();
   }
 }

运行结果:

深入了解c# 信号量和互斥体

从运行结果看出两个进程之间的时间间隔为5秒,当我们把构造函数中命名参数去掉时就可以看出差别了。

到这里多线程处理基本上讲完,这个系列也只是一个入门,真真要好好掌握多线程,还是要在项目中多去实战的。接下来我可能会做一个小的例子的,大概的思路是实现一个文件的下载的这样的例子。如果大家有什么好的例子来运用多线程的知识的话,可以留言给我,我也会尽量去实现(如果不会的话,这样也可以促使我去学习),实现后也会和大家分享的。

来源:https://www.kancloud.cn/wizardforcel/learning-hard-csharp/111531

标签:c#,信号量,互斥体
0
投稿

猜你喜欢

  • Java 面试题基础知识集锦

    2022-05-25 00:01:58
  • Java设计模式之建造者模式

    2023-09-03 14:40:37
  • WPF+ASP.NET SignalR实现简易在线聊天功能的示例代码

    2023-01-09 09:34:07
  • C#实现文本文件读写方法汇总

    2023-10-27 16:57:40
  • 如何用Stream解决两层List属性求和问题

    2022-07-31 20:32:35
  • Java多线程下载网图的完整案例

    2022-10-17 23:59:02
  • Java如何去掉指定字符串的开头的指定字符

    2022-05-23 13:28:31
  • Spring如何更简单的读取和存储对象

    2023-09-30 00:32:14
  • java list,set,map,数组间的相互转换详解

    2023-04-11 13:02:04
  • 基于Java的打包jar、war、ear包的作用与区别详解

    2023-11-17 11:41:13
  • spring-boot-maven-plugin报红解决方案(亲测有效)

    2022-07-23 01:16:46
  • Spring Boot详解配置文件的用途与用法

    2022-08-01 20:37:27
  • java编写的文件管理器代码分享

    2023-11-20 13:28:52
  • Android实现美团外卖底部导航栏动画

    2022-09-21 20:09:13
  • SpringBoot过滤器的使用

    2023-08-28 21:28:56
  • java实现gif动画效果(java显示动态图片)

    2023-10-19 07:56:38
  • Java开发工具IntelliJ IDEA安装图解

    2022-06-14 02:30:20
  • Java 随机生成验证码(支持大小写字母、数字、随机字体)的实例

    2023-11-25 00:35:38
  • Java 数据结构与算法系列精讲之时间复杂度与空间复杂度

    2022-03-19 20:19:50
  • 在Spring Boot中加载初始化数据的实现

    2023-08-18 10:01:44
  • asp之家 软件编程 m.aspxhome.com