深入了解c# 信號(hào)量和互斥體
一、信號(hào)量(Semaphore)
信號(hào)量(Semaphore)是由內(nèi)核對(duì)象維護(hù)的int變量,當(dāng)信號(hào)量為0時(shí),在信號(hào)量上等待的線程會(huì)堵塞,信號(hào)量大于0時(shí),就解除堵塞。當(dāng)在一個(gè)信號(hào)量上等待的線程解除堵塞時(shí),內(nèi)核自動(dòng)會(huì)將信號(hào)量的計(jì)數(shù)減1。在.net 下通過(guò)Semaphore類來(lái)實(shí)現(xiàn)信號(hào)量同步。
Semaphore類限制可同時(shí)訪問(wèn)某一資源或資源池的線程數(shù)。WaitOne method, which is inherited from the WaitHandle class, and release the semaphore by calling the Release method.">線程通過(guò)調(diào)用 WaitOne方法將信號(hào)量減1,并通過(guò)調(diào)用 Release方法把信號(hào)量加1。
先說(shuō)下構(gòu)造函數(shù):
public Semaphore(int initialCount,int maximumCount);通過(guò)兩個(gè)參數(shù)來(lái)設(shè)置信號(hào)的初始計(jì)數(shù)和最大計(jì)數(shù)。
下面通過(guò)一段代碼來(lái)演示信號(hào)量同步的使用:
class Program { // 初始信號(hào)量計(jì)數(shù)為0,最大計(jì)數(shù)為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)); // 開(kāi)始線程,并傳遞參數(shù) test.Start(i); } // 等待1秒讓所有線程開(kāi)始并阻塞在信號(hào)量上 Thread.Sleep(500); // 信號(hào)量計(jì)數(shù)加4 // 最后可以看到輸出結(jié)果次數(shù)為4次 semaphore.Release(4); Console.Read(); } public static void TestMethod(object number) { // 設(shè)置一個(gè)時(shí)間間隔讓輸出有順序 int span = Interlocked.Add(ref time, 100); Thread.Sleep(1000 + span); //信號(hào)量計(jì)數(shù)減1 semaphore.WaitOne(); Console.WriteLine("Thread {0} run ", number); } }
運(yùn)行結(jié)果:
同樣信號(hào)量也可以實(shí)現(xiàn)進(jìn)程中線程的同步,同樣也是通過(guò)對(duì)信號(hào)量命名來(lái)實(shí)現(xiàn)的,
通過(guò)調(diào)用public Semaphore(int initialCount,int maximumCount,string name);該構(gòu)造函數(shù)多傳入一個(gè)信號(hào)量名來(lái)實(shí)現(xiàn)
下面一段實(shí)例代碼來(lái)演示下:
using System; using System.Threading; namespace SemaphoreSample { class Program { // 初始信號(hào)量計(jì)數(shù)為4,最大計(jì)數(shù)為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)); // 開(kāi)始線程,并傳遞參數(shù) test.Start(i); } // 等待1秒讓所有線程開(kāi)始并阻塞在信號(hào)量上 Thread.Sleep(1000); Console.Read(); } public static void TestMethod(object number) { // 設(shè)置一個(gè)時(shí)間間隔讓輸出有順序 int span = Interlocked.Add(ref time, 500); Thread.Sleep(1000 + span); //信號(hào)量計(jì)數(shù)減1 semaphore.WaitOne(); Console.WriteLine("Thread {0} run ", number); } } }
運(yùn)行結(jié)果:
從運(yùn)行結(jié)果中可以看出, 第二個(gè)進(jìn)程值運(yùn)行了一行語(yǔ)句, 因?yàn)槲覀冊(cè)O(shè)置的初始信號(hào)計(jì)數(shù)為4,每運(yùn)行一個(gè)線程,信號(hào)計(jì)數(shù)通過(guò)調(diào)用WaitOne方法減1,所以第二個(gè)進(jìn)行一開(kāi)始信號(hào)計(jì)數(shù)為1而不是進(jìn)程一中的4,如果我們把信號(hào)計(jì)數(shù)后面的name參數(shù)去除的話,此時(shí)第二個(gè)進(jìn)程和第一個(gè)進(jìn)程中的結(jié)果應(yīng)該是一樣的(因?yàn)榇藭r(shí)沒(méi)有進(jìn)行不同進(jìn)程中線程的同步)。
二、互斥體(Mutex)
同樣互斥體也是同樣可以實(shí)現(xiàn)線程之間的同步和不同進(jìn)程中線程的同步的
先看看線程之間的同步的例子吧(在這里我也不多做解釋了,因?yàn)樗麄冎g的使用很類似,直接貼出代碼):
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); // 開(kāi)始線程,并傳遞參數(shù) test.Start(); } Console.Read(); } public static void TestMethod() { mutex.WaitOne(); Thread.Sleep(500); count++; Console.WriteLine("Current Cout Number is {0}", count); mutex.ReleaseMutex(); } }
運(yùn)行結(jié)果:
實(shí)現(xiàn)進(jìn)程間同步:
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(); } }
運(yùn)行結(jié)果:
從運(yùn)行結(jié)果看出兩個(gè)進(jìn)程之間的時(shí)間間隔為5秒,當(dāng)我們把構(gòu)造函數(shù)中命名參數(shù)去掉時(shí)就可以看出差別了。
到這里多線程處理基本上講完,這個(gè)系列也只是一個(gè)入門,真真要好好掌握多線程,還是要在項(xiàng)目中多去實(shí)戰(zhàn)的。接下來(lái)我可能會(huì)做一個(gè)小的例子的,大概的思路是實(shí)現(xiàn)一個(gè)文件的下載的這樣的例子。如果大家有什么好的例子來(lái)運(yùn)用多線程的知識(shí)的話,可以留言給我,我也會(huì)盡量去實(shí)現(xiàn)(如果不會(huì)的話,這樣也可以促使我去學(xué)習(xí)),實(shí)現(xiàn)后也會(huì)和大家分享的。
以上就是深入了解c# 信號(hào)量和互斥體的詳細(xì)內(nèi)容,更多關(guān)于c# 信號(hào)量和互斥體的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
WPF開(kāi)發(fā)txt閱讀器實(shí)現(xiàn)目錄提取功能
這篇文章主要為大家詳細(xì)介紹了WPF開(kāi)發(fā)txt閱讀器時(shí)如何實(shí)現(xiàn)目錄提取功能,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-06-06C#序列化與反序列化集合對(duì)象并進(jìn)行版本控制
這篇文章介紹了C#序列化與反序列化集合對(duì)象并實(shí)現(xiàn)版本控制的方法,文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-09-09c# 冒泡排序算法(Bubble Sort) 附實(shí)例代碼
這篇文章主要介紹了c# 冒泡排序算法,需要的朋友可以參考下2013-10-10C#類型轉(zhuǎn)換之自定義隱式轉(zhuǎn)換和顯式轉(zhuǎn)換
本文主要為大家介紹了一個(gè)新的類型轉(zhuǎn)換方法:通過(guò)自定義隱式轉(zhuǎn)換,把不一樣的數(shù)據(jù)類型反序列化為一樣的數(shù)據(jù)類型,需要的同學(xué)可以參考一下2022-03-03Unity?UGUI的StandaloneInputModule標(biāo)準(zhǔn)輸入模塊組件使用示例
這篇文章主要為大家介紹了Unity?UGUI的StandaloneInputModule標(biāo)準(zhǔn)輸入模塊組件使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-08-08C#切換鼠標(biāo)左右鍵習(xí)慣無(wú)需控制面板中修改
本人一直喜歡左手使用鼠標(biāo),偶爾同事會(huì)臨時(shí)操作一下,因?yàn)樗牧?xí)慣是右手,還得在控制面板里進(jìn)行更改,太麻煩了所以就編寫一個(gè)控制臺(tái)程序,雙擊一下即可切換左右鍵,熱愛(ài)懶人的你可不要錯(cuò)過(guò)了哈2013-02-02C# Winform 讓整個(gè)窗口都可以拖動(dòng)
Windows 的 API 果然強(qiáng)大啊.以前要實(shí)現(xiàn)全窗口拖動(dòng), 要寫鼠標(biāo)按下和抬起事件, 很是麻煩, 偶爾還會(huì)出現(xiàn) BUG2011-05-05