C#多線程--信號量(Semaphore)


百度百科:Semaphore,是負責協調各個線程, 以保證它們能夠正確、合理的使用公共資源。也是操作系統中用於控制進程同步互斥的量。

Semaphore常用的方法有兩個WaitOne()Release(),Release()的作用是退出信號量並返回前一個計數,而WaitOne()則是阻止當前線程,直到當前線程的WaitHandle 收到信號。這里我舉一個例子讓大家更容易理解:當我們這樣實例化Semaphore時候

Semaphore sema = new Semaphore(x,y)

有一隊人排隊上洗手間,人就相當於線程,x為還剩余的位置數量,y為總的位置數量。

WaitOne()方法就相當於人在等待洗手間位置的行為,而Release()方法就相當於一個人從洗手間出來的行為,這里再假設x和y都為5,說明開始的時候洗手間有5個空位置,且總共只有5個位置,當一隊超過5個人的隊伍要上洗手間的就排隊,首先WaitOne()方法等待,發現有空位就依次進去,每進去一個空位減一,直到進去5之后個沒有空位,這時候后面的人就一直等待,直到進去的人從洗手間出來Release()方法,空位加一,在等待WaitOne()方法的人發現有空位又進去一個空位減一……如此循環往復。

請看下面例子:

public class Program
    {
        static Semaphore sema = new Semaphore(5, 5);
        const int cycleNum = 9;
        static void Main(string[] args) 
        {
            for(int i = 0; i < cycleNum; i++)
            {
                Thread td = new Thread(new ParameterizedThreadStart(testFun));
                td.Name = string.Format("編號{0}",i.ToString());
                td.Start(td.Name);
            }
            Console.ReadKey();
        }
        public static void testFun(object obj)
        {
            sema.WaitOne();
            Console.WriteLine(obj.ToString() + "進洗手間:" + DateTime.Now.ToString());
            Thread.Sleep(2000);
            Console.WriteLine(obj.ToString() + "出洗手間:" + DateTime.Now.ToString());
            sema.Release();
        }
    }

運行結果如下:

image

這里我要說明一點,信號量控制的只是線程同步的量,而不管順序,這個例子來說線程控制的就是線程同步量為5,也就是同時並發的線程數量為5個,至於是哪個先哪個后不是由這里的信號量決定的。

當然這個例子中因沒有做什么復雜的操作,一般情況進入線程的時間和每個線程要的時間不會有太大差別,所以執行的順序還是很規律的(為了說明這個問題我也是運行了多次才讓結果稍有不同,這里編號2搶在了編號1前面就是這個道理),如果線程中有很復雜的操作每個線程在運行中所用的時間有比較大的差別,或者循環開始有復雜操作那么很可能就不是編號0先進入洗手間了,且不一定是先進入的就會先出來。

接下來再簡單介紹一下Semaphore的WaitOne()和Release()的重載方法

public int Release(int releaseCount);

releaseCount指的是釋放的信號量數量,如果沒有參數默認為1,Release()就相當於Release(1)

這里要說明一點,當Release()或者Release(int releaseCount)執行時導致信號量計數大於最大數量時會拋出SemaphoreFullException異常。下面這種情況就會異常:

Semaphore sem = new Semaphore(4,5);
sem.Release(2);//這里是釋放2個信號量加上之前的4個,超出5個了

public virtual bool WaitOne(TimeSpan timeout);
public virtual bool WaitOne(int millisecondsTimeout);

第一個重載參數timeout:指定時間間隔,若在這段時間內沒有接收到信號則跳過等待繼續執行

第二個重載參數millisecondsTimeout:指定時間間隔整數毫秒,若在這段時間內沒有接收到信號則跳過等待繼續執行

WaitOne()還有兩個重載方法不是很常用這里就不介紹了。上面的重載方法這里也不再進了案例說明了,有興趣的朋友可以自己嘗試一下。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM