使用SemaphoreSlim类(作为Semaphore类的轻量级版本)控制同时访问同一资源的线程数量

using System;
using System.Threading;

namespace ThreadDemo_SemaphoreSlim
{
    class Program
    {
        static SemaphoreSlim _semaphore = new SemaphoreSlim(4);

        static void AccessDatabase(string name, int seconds)
        {
            Console.WriteLine("{0} waits to access a database", name);
            _semaphore.Wait();
            Console.WriteLine("{0} was granted an access to a database", name);
            Thread.Sleep(TimeSpan.FromSeconds(seconds));
            Console.WriteLine("{0} is completed!", name);
            _semaphore.Release();
        }
        static void Main(string[] args)
        {
            for(int i = 1; i <= 6; i++)
            {
                string threadName = "Thread " + i;
                int secondsToWait = 2 + 2 * i;
                var t = new Thread(() => AccessDatabase(threadName, secondsToWait));
                t.Start();
            }
            Console.ReadKey();
        }
    }
}

    当主程序启动时,创建了SemaphoreSlim的一个实例,并在其构造函数中指定允许的并发线程数量。然后启动6个不同名称和不同初始运行时间的线程。

    每个线程都尝试获取数据库的访问,但是我们借助于信号系统限制了访问数据库的并发数为4个线程。当有4个线程获取了数据库的访问后,其他两个线程需要等待,直到之前线程中的某一个完成工作并调用_semaphore.Release()方法来发出信号。

     此类并不支持进程间同步,所以跨程序同步的场景下可以使用Semaphore.

运行结果:

使用SemaphoreSlim类(作为Semaphore类的轻量级版本)控制同时访问同一资源的线程数量