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

一、信号量(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);
    }
  }

运行结果:

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

通过调用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);
    }
  }
}

运行结果:

从运行结果中可以看出, 第二个进程值运行了一行语句, 因为我们设置的初始信号计数为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();
    }
  }

运行结果:

实现进程间同步:

 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();
    }
  }

运行结果:

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

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

以上就是深入了解c# 信号量和互斥体的详细内容,更多关于c# 信号量和互斥体的资料请关注我们其它相关文章!

(0)

相关推荐

  • 详细分析c# 运算符重载

    您可以重定义或重载 C# 中内置的运算符.因此,程序员也可以使用用户自定义类型的运算符.重载运算符是具有特殊名称的函数,是通过关键字 operator 后跟运算符的符号来定义的.与其他函数一样,重载运算符有返回类型和参数列表. 例如,请看下面的函数: public static Box operator+ (Box b, Box c) { Box box = new Box(); box.length = b.length + c.length; box.breadth = b.breadth

  • C# 引入委托的目的是什么

    引言: 对于一些刚接触C# 不久的朋友可能会对C#中一些基本特性理解的不是很深,然而这些知识也是面试时面试官经常会问到的问题,所以我觉得有必要和一些接触C#不久的朋友分享下关于C#基础知识的文章,所以有了这个系列,希望通过这个系列让朋友对C#的基础知识理解能够更进一步.然而委托又是C#基础知识中比较重要的一点,基本上后面的特性都和委托有点关系,所以这里就和大家先说说委托,为什么我们需要委托. 一.C#委托是什么的? 在正式介绍委托之前,我想下看看生活中委托的例子--生活中,如果如果我们需要打官司

  • C#信号量用法简单示例

    本文实例讲述了C#信号量用法.分享给大家供大家参考,具体如下: using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading; /* * 标题:如何使用信号量的示例代码 * Author:kagula * Date:2015-6-16 * Environment:VS2010SP1, .NET Framework 4 client profi

  • c# 委托的本质是什么

    引言 上一个专题已经和大家分享了我理解的--C#中为什么需要委托,专题中简单介绍了下委托是什么以及委托简单的应用的,在这个专题中将对委托做进一步的介绍的,本专题主要对委本质和委托链进行讨论. 一.委托的本质 平时我们很容易使用委托--用C# delegate关键字定义委托,再用new操作符构造委托实例,然后通过调用委托实例来调用回调方法(就是用一个了委托对象的变量来代替方法名,这句话如果刚接触的人不好理解的话,这里给个例子:MyDelegate mydelegate =new Mydelegat

  • 详解c# 线程同步

    一.线程同步概述 前面的文章都是讲创建多线程来实现让我们能够更好的响应应用程序,然而当我们创建了多个线程时,就存在多个线程同时访问一个共享的资源的情况,在这种情况下,就需要我们用到线程同步,线程同步可以防止数据(共享资源)的损坏. 然而我们在设计应用程序还是要尽量避免使用线程同步, 因为线程同步会产生一些问题: 1. 它的使用比较繁琐.因为我们要用额外的代码把多个线程同时访问的数据包围起来,并获取和释放一个线程同步锁,如果我们在一个代码块忘记获取锁,就有可能造成数据损坏. 2. 使用线程同步会影

  • C#中的虚方法和抽象方法的运用

    今天在云和学院学了很多,我这次只能先总结一下C#中的虚方法和抽象的运用. 理论: 虚方法: 用virtual修饰的方法叫做虚方法 虚方法可以在子类中通过override关键字来重写 常见的虚方法:ToString() Equals 抽象方法: 抽象类与抽象方法由abstract修饰 abstract的使用注意 –抽象方法没有方法体 –抽象成员只能存在于抽象类中 –抽象类可以有非抽象成员 –抽象类的派生类必须实现抽象方法体 –抽象类只能用作基类,无法实例化 补充: 重写:不同对象对于同一个方法调用

  • C#获取本地IP的四种方式示例详解

    1.第一种方式 采用System.Net.Dns的GetHostAddress的方式,具体请看代码: /// <summary> /// 网络不通畅可以获取 /// 不过能获取到具体的IP /// </summary> /// <returns></returns> public static List<IPAddress> GetByGetHostAddresses() { try { IPAddress[] adds = Dns.GetHos

  • c# 实现位图算法(BitMap)

    算法原理 BitMap的基本思想就是用一个bit位来标记某个元素对应的Value,而Key即是该元素.由于采用了Bit为单位来存储数据,因此可以大大节省存储空间. BitMap可以看成一种数据结构. 假设有这样一个需求:在20亿个随机整数中找出某个数m是否存在其中,并假设32位操作系统,4G内存. 在Java中,int占4字节,1字节=8位(1 byte = 8 bit). 如果每个数字用int存储,那就是20亿个int,因而占用的空间约为 (2000000000*4/1024/1024/102

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

    一.信号量(Semaphore) 信号量(Semaphore)是由内核对象维护的int变量,当信号量为0时,在信号量上等待的线程会堵塞,信号量大于0时,就解除堵塞.当在一个信号量上等待的线程解除堵塞时,内核自动会将信号量的计数减1.在.net 下通过Semaphore类来实现信号量同步. Semaphore类限制可同时访问某一资源或资源池的线程数.WaitOne method, which is inherited from the WaitHandle class, and release t

  • 易语言通过句柄名来关闭互斥体实现游戏多开的代码

    DLL命令代码 .版本 2 .DLL命令 GetProcessHandleCount, 逻辑型 .参数 hProcess, 整数型 .参数 pdwHandleCount, 整数型, 传址 .DLL命令 DuplicateHandle, 逻辑型 .参数 hSourceProcessHandle, 整数型 .参数 hSourceHandle, 整数型 .参数 hTargetProcessHandle, 整数型 .参数 lpTargetHandle, 整数型, 传址 .参数 dwDesiredAcce

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

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

  • c#互斥锁Mutex类用法介绍

    什么是Mutex “mutex”是术语“互相排斥(mutually exclusive)”的简写形式,也就是互斥量.互斥量跟临界区中提到的Monitor很相似,只有拥有互斥对象的线程才具有访问资源的权限,由于互斥对象只有一个,因此就决定了任何情况下此共享资源都不会同时被多个线程所访问.当前占据资源的线程在任务处理完后应将拥有的互斥对象交出,以便其他线程在获得后得以访问资源.互斥量比临界区复杂,因为使用互斥不仅仅能够在同一应用程序不同线程中实现资源的安全共享,而且可以在不同应用程序的线程之间实现对

  • FreeRTOS进阶信号量示例的完全解析

    目录 FreeRTOS的信号量包括二进制信号量.计数信号量.互斥信号量(以后简称互斥量)和递归互斥信号量(以后简称递归互斥量).关于它们的区别可以参考< FreeRTOS系列第19篇---FreeRTOS信号量>一文. 信号量API函数实际上都是宏,它使用现有的队列机制.这些宏定义在semphr.h文件中.如果使用信号量或者互斥量,需要包含semphr.h头文件. 二进制信号量.计数信号量和互斥量信号量的创建API函数是独立的,但是获取和释放API函数都是相同的:递归互斥信号量的创建.获取和释

  • Linux多线程使用互斥量同步线程

    本文将会给出互斥量的详细解说,并用一个互斥量解决上一篇文章中,要使用两个信号量才能解决的只有子线程结束了对输入的处理和统计后,主线程才能继续执行的问题. 一.什么是互斥量 互斥量是另一种用于多线程中的同步访问方法,它允许程序锁住某个对象,使得每次只能有一个线程访问它.为了控制对关键代码的访问,必须在进入这段代码之前锁住一个互斥量,然后在完成操作之后解锁. 二.互斥量的函数的使用 它们的定义与使用信号量的函数非常相似,它们的定义如下: #include <pthread.h> int pthre

  • c# mutex互斥量的深入解析

    互斥锁(Mutex) 互斥锁是一个互斥的同步对象,意味着同一时间有且仅有一个线程可以获取它. 互斥锁可适用于一个共享资源每次只能被一个线程访问的情况  函数: //创建一个处于未获取状态的互斥锁 Public Mutex(); //如果owned为true,互斥锁的初始状态就是被主线程所获取,否则处于未获取状态 Public Mutex(bool owned); 如果要获取一个互斥锁.应调用互斥锁上的WaitOne()方法,该方法继承于Thread.WaitHandle类 它处于等到状态直至所调

  • python信号量,条件变量和事件详解

    一.在线程中获取时间,判断当前时间三面之后,触发“事件”对象. 运行结果: 二.在另一个线程中,作为数学考试结束的判断变量,否则一直处于考试之中,并打印.  运行结果: 什么是信号量: 互斥锁 同时只允许一个线程更改数据,而Semaphore是同时允许一定数量的线程更改数据 ,比如厕所有3个坑,那最多只允许3个人上厕所,后面的人只能等里面有人出来了才能再进去. 什么是条件变量: Python提供的Condition对象提供了对复杂线程同步问题的支持.Condition被称为条件变量,除了提供与L

  • FreeRTOS实时操作系统信号量基础

    目录 前言 1.信号量简介 2.二进制信号量 3.计数信号量 4.互斥量 5.递归互斥量 前言 本文介绍信号量的基础知识,详细源码分析见<FreeRTOS进阶FreeRTOS信号量分析> 1.信号量简介 FreeRTOS的信号量包括二进制信号量.计数信号量.互斥信号量(以后简称互斥量)和递归互斥信号量(以后简称递归互斥量). 我们可以把互斥量和递归互斥量看成特殊的信号量.互斥量和信号量在用法上不同: 信号量用于同步,任务间或者任务和中断间同步:互斥量用于互锁,用于保护同时只能有一个任务访问的资

  • FreeRTOS信号量API函数基础教程

    目录 前言 1创建二进制信号量 1.1函数描述 2创建计数信号量 3创建互斥量 4创建递归互斥量 5删除信号量 6获取信号量 7获取信号量(带中断保护) 8获取递归互斥量 9释放信号量 10释放信号量(带中断保护) 11释放递归互斥量 12获取互斥量持有任务的句柄 前言 FreeRTOS的信号量包括二进制信号量.计数信号量.互斥信号量(以后简称互斥量)和递归互斥信号量(以后简称递归互斥量).我们可以把互斥量和递归互斥量看成特殊的信号量. 信号量API函数实际上都是宏,它使用现有的队列机制.这些宏

随机推荐