Где используются семафоры и мьютексы?

Недавно узнал что есть такие штуки в параллельном программировании как семафоры и мьютексы. Расскажите пожалуйста, какое применение они нашли в языке C#? Где чаще всего их используют, и зачем?

Вот, почитайте этот пост, тут уже похожая тема была рассмотрена, найдёте для себя немало полезного - https://forum.itvdn.com/thread/vse-pro-simaforu-v-c/.

Вот вам пример использования мьютексов (семафоры рассмотрены в посте, что я кинул чуть выше) :

// Применить мьютекс.
using System;
using System.Threading;
//В этом классе содержится общий ресурс(переменная Count),
// а также мьютекс (Mtx), управляющий доступом к ней.
class SharedRes {
  public static int Count = 0;
  public static Mutex Mtx = new Mutex();
}
// В этом потоке переменная SharedRes.Count инкрементируется,
class IncThread {
  int num;
  public Thread Thrd;
  public IncThread(string name, int n) {
    Thrd = new Thread(this.Run);
    num = n;
    Thrd.Name = name;
    Thrd.Start();
  }
  // Точка входа в поток,
  void Run() {
    Console.WriteLine(Thrd.Name   " ожидает мьютекс.");
    // Получить мьютекс.
    SharedRes.Mtx.WaitOne();
    Console.WriteLine(Thrd.Name   " получает мьютекс.");
    do {
      Thread.Sleep (500);
      SharedRes.Count  ;
      Console.WriteLine("В потоке "   Thrd.Name  
           ", SharedRes.Count = "   SharedRes.Count);
      num -- ;
    } while(num > 0);
    Console.WriteLine(Thrd.Name   " освобождает мьютекс.");
    // Освободить мьютекс.
    SharedRes.Mtx.ReleaseMutex();
  }
}
// В этом потоке переменная SharedRes.Count декрементируется,
class DecThread {
  int num;
  public Thread Thrd;
  public DecThread(string name, int n) {
    Thrd = new Thread(new ThreadStart(this.Run));
    num = n;
    Thrd.Name = name;
    Thrd.Start();
  }
  // Точка входа в поток,
  void Run() {
    Console.WriteLine(Thrd.Name   " ожидает мьютекс.");
    // Получить мьютекс.
    SharedRes.Mtx.WaitOne();
    Console.WriteLine(Thrd.Name   " получает мьютекс.");
    do {
      Thread.Sleep(500) ;
      SharedRes.Count--;
      Console.WriteLine("В потоке "   Thrd.Name  
             ", SharedRes.Count = "   SharedRes.Count);
      num-- ;
    } while(num > 0);
    Console.WriteLine(Thrd.Name   " освобождает мьютекс.");
    // Освободить мьютекс.
    SharedRes.Mtx.ReleaseMutex();
  }
}
class MutexDemo {
  static void Main() {
    // Сконструировать два потока.
    IncThread mt1 = new IncThread("Инкрементирующий Поток", 5);
    Thread.Sleep(1); // разрешить инкрементирующему потоку начаться
    DecThread mt2 = new DecThread("Декрементирующий Поток", 5);
    mt1.Thrd.Join();
    mt2.Thrd.Join();
  }
}