C# 多线程编程:线程锁与无锁并发

文章目录

  • 前言
  • 一、锁的基本概念
    • 1.1 什么是锁?
    • 1.2 为什么需要锁?
    • 1.3 锁的作用原理
  • 二、线程锁的类型
    • 2.1 自旋锁(Spin Lock)
    • 2.2 互斥锁(Mutex)
    • 2.3 混合锁(Hybrid Lock)
    • 2.4 读写锁(Read-Write Lock)
  • 三、锁的实现方式
    • 3.1 Monitor(互斥体)
    • 3.2 Mutex(互斥体)
    • 3.3 Semaphore(信号量)
    • 3.4 ReaderWriterLock(读写锁)
  • 四、无锁并发编程
    • 4.1 无锁并发编程的概念
    • 4.2 无锁算法
      • 4.2.1 CAS(Compare And Swap)
      • 4.2.2 Volatile 关键字
    • 4.3 无锁并发编程的优势
    • 4.4 无锁并发编程的局限性
  • 五、并发集合类
    • 5.1 ConcurrentBag
    • 5.2 ConcurrentDictionary
    • 5.3 ConcurrentQueue
    • 5.4 ConcurrentStack
  • 六、经典并发同步问题
    • 6.1 生产者-消费者问题(Producer-Consumer Problem)
      • 6.1.1 使用 `Monitor` 类实现生产者-消费者问题
      • 6.1.2 使用 `Semaphore` 类实现生产者-消费者问题
      • 6.1.3 使用 `BlockingCollection` 类实现生产者-消费者问题
    • 6.2 读者-写者问题(Reader-Writer Problem)
      • 6.2.1 使用 `ReaderWriterLockSlim` 类实现读者-写者问题
      • 6.2.2 使用 `SemaphoreSlim` 类实现读者-写者问题
      • 6.2.3 使用 `Monitor` 类实现读者-写者问题
    • 6.3 哲学家就餐问题(Dining Philosophers Problem)
      • 6.3.1 使用`Semaphore`实现哲学家就餐问题
      • 6.3.2 使用`Mutex`实现哲学家就餐问题
      • 6.3.3 使用`Monitor`实现哲学家就餐问题
  • 总结


前言

多线程编程在现代软件开发中至关重要。本文将讨论 C# 中的多线程技术,重点介绍锁的概念,线程锁与无锁并发。通过学习本篇博文,我们将学会如何正确处理并发问题,提高程序的性能和稳定性。


一、锁的基本概念

在多线程编程中,掌握锁的概念至关重要。本节将介绍什么是锁,为什么我们需要锁以及锁的作用原理。

1.1 什么是锁?

锁是一种同步机制,用于控制多个线程对共享资源的访问。当一个线程获得了锁时,其他线程将被阻塞,直到该线程释放了锁。

1.2 为什么需要锁?

在并发编程中,多个线程同时访问共享资源可能导致数据竞争和不确定的行为。锁可以确保在任意时刻只有一个线程可以访问共享资源,从而避免竞态条件和数据不一致性问题。

1.3 锁的作用原理

锁的作用原理通常涉及到内部的互斥机制。当一个线程获得锁时,它会将锁标记为已被占用,其他线程尝试获取该锁时会被阻塞,直到持有锁的线程释放锁。这种互斥机制可以通过不同的算法和数据结构来实现,如互斥量、自旋锁等。

理解锁的概念是进行多线程编程的基础,它为我们提供了一种可靠的方式来保护共享资源,确保线程安全和程序的正确性。在接下来的章节中,我们将深入探讨不同类型的锁以及它们在 C# 多线程编程中的应用。

二、线程锁的类型

在多线程编程中,锁的实现通常基于互斥机制,确保在任意时刻只有一个线程可以访问共享资源。本节将介绍几种常见的锁类型,包括自旋锁、互斥锁、混合锁和读写锁。

2.1 自旋锁(Spin Lock)

  • 自旋锁是一种基于忙等待的锁,当线程尝试获取锁时,如果发现锁已被其他线程占用,它会循环(自旋)等待,不断地检查锁是否被释放。
  • 自旋锁适用于锁的占用时间短、线程并发度高的情况,因为它避免了线程在等待锁时进入内核态造成的性能损失。
  • 但自旋锁可能会导致线程空转消耗 CPU 资源,因此不适合在锁被占用时间较长或竞争激烈的情况下使用。

2.2 互斥锁(Mutex)

  • 互斥锁是一种阻塞式锁,它通过操作系统提供的原语实现,当线程尝试获取锁时,如果发现锁已被其他线程占用,它会被阻塞,直到锁被释放。
  • 互斥锁适用于锁的占用时间长、线程竞争激烈的情况,因为它可以将等待锁的线程置于休眠状态,避免空转浪费 CPU 资源。
  • 但互斥锁由于涉及系统调用,因此会产生较大的开销,尤其在高并发情况下可能成为性能瓶颈。

2.3 混合锁(Hybrid Lock)

  • 混合锁是结合了自旋锁和互斥锁的优点,根据锁的占用情况动态选择使用自旋等待还是阻塞等待。
  • 在锁的竞争不激烈时,混合锁会采用自旋等待的方式,避免线程进入内核态;而在锁的竞争激烈时,会转为阻塞等待,以减少空转和CPU资源的浪费。
  • 混合锁的实现较为复杂,需要根据具体的场景进行调优,以达到最佳的性能和资源利用率。

2.4 读写锁(Read-Write Lock)

  • 读写锁允许多个线程同时对共享资源进行读取操作,但在进行写入操作时需要互斥。
  • 读写锁适用于读操作远远多于写操作的场景,可以提高程序的并发性能。
  • 读写锁通常包含一个写锁和多个读锁,当写锁被占用时,所有的读锁和写锁都会被阻塞;而当读锁被占用时,其他的读锁仍然可以被获取,但写锁会被阻塞。

三、锁的实现方式

下面是几种常见的锁类型:

3.1 Monitor(互斥体)

Monitor 是 C# 中最基本的锁机制之一,它使用 lock 关键字来实现。lock 关键字在进入代码块时获取锁,在退出代码块时释放锁。这确保了在同一时刻只有一个线程可以执行 lock 块中的代码。

using System;
using System.Threading;class Program
{private static object _lock = new object();static void Main(string[] args){// 启动两个线程访问临界区Thread thread1 = new Thread(EnterCriticalSection);Thread thread2 = new Thread(EnterCriticalSection);thread1.Start();thread2.Start();}static void EnterCriticalSection(){// 进入临界区Monitor.Enter(_lock);try{// 在临界区内操作共享资源Console.WriteLine($"Thread {Thread.CurrentThread.ManagedThreadId} entered critical section.");Thread.Sleep(2000);}finally{// 退出临界区Monitor.Exit(_lock);Console.WriteLine($"Thread {Thread.CurrentThread.ManagedThreadId} exited critical section.");}}
}

另一种写法:

object lockObj = new object();
lock (lockObj)
{// 执行需要同步的代码
}

3.2 Mutex(互斥体)

Mutex 是一种操作系统级别的同步原语,与 Monitor 不同,Mutex 可以在进程间共享。Mutex 是一个系统对象,它可以在全局范围内唯一标识一个锁。使用 Mutex 需要在代码中声明一个 Mutex 对象,然后通过 WaitOne 和 ReleaseMutex 方法来获取和释放锁。

using System;
using System.Threading;class Program
{private static Mutex _mutex = new Mutex();static void Main(string[] args){// 启动两个线程访问临界区Thread thread1 = new Thread(EnterCriticalSection);Thread thread2 = new Thread(EnterCriticalSection);thread1.Start();thread2.Start();}static void EnterCriticalSection(){// 等待获取 Mutex_mutex.WaitOne();try{// 在临界区内操作共享资源Console.WriteLine($"Thread {Thread.CurrentThread.ManagedThreadId} entered critical section.");Thread.Sleep(2000);}finally{// 释放 Mutex_mutex.ReleaseMutex();Console.WriteLine($"Thread {Thread.CurrentThread.ManagedThreadId} exited critical section.");}}
}

3.3 Semaphore(信号量)

Semaphore 是一种允许多个线程同时访问共享资源的同步原语。它通过一个计数器来控制同时访问资源的线程数量。Semaphore 构造函数需要指定初始的计数器值和最大的计数器值。通过 WaitOne 和 Release 方法来获取和释放信号量。

using System;
using System.Threading;class Program
{private static Semaphore _semaphore = new Semaphore(2, 2); // 允许最多两个线程同时访问static void Main(string[] args){// 启动五个线程访问临界区for (int i = 0; i < 5; i++){Thread thread = new Thread(EnterCriticalSection);thread.Start(i);}}static void EnterCriticalSection(object threadId){// 等待获取 Semaphore_semaphore.WaitOne();try{// 在临界区内操作共享资源Console.WriteLine($"Thread {threadId} entered critical section.");Thread.Sleep(2000);}finally{// 释放 Semaphore_semaphore.Release();Console.WriteLine($"Thread {threadId} exited critical section.");}}
}

3.4 ReaderWriterLock(读写锁)

ReaderWriterLock 是一种特殊的锁机制,它允许多个线程同时读取共享资源,但在写入资源时需要互斥。这种锁适用于读操作远远多于写操作的场景,可以提高性能。

using System;
using System.Threading;class Program
{private static ReaderWriterLockSlim _rwLock = new ReaderWriterLockSlim();static void Main(string[] args){// 启动五个读线程和一个写线程访问共享资源for (int i = 0; i < 5; i++){Thread readerThread = new Thread(ReadSharedResource);readerThread.Start(i);}Thread writerThread = new Thread(WriteSharedResource);writerThread.Start();}static void ReadSharedResource(object threadId){_rwLock.EnterReadLock();try{// 读取共享资源Console.WriteLine($"Reader {threadId} read shared resource.");Thread.Sleep(2000);}finally{_rwLock.ExitReadLock();}}static void WriteSharedResource(){_rwLock.EnterWriteLock();try{// 写入共享资源Console.WriteLine("Writer wrote shared resource.");Thread.Sleep(1000);}finally{_rwLock.ExitWriteLock();}}
}

四、无锁并发编程

在多线程编程中,除了使用锁机制来保护共享资源外,还可以通过无锁并发编程来实现并发控制。本章将介绍无锁并发编程的概念、优势以及常见的无锁算法。

4.1 无锁并发编程的概念

无锁并发编程是一种基于原子操作的并发控制方式,它不需要使用传统的锁机制来保护共享资源,而是通过原子性操作来确保线程安全。无锁并发编程通常比锁机制具有更低的开销和更高的性能。

4.2 无锁算法

4.2.1 CAS(Compare And Swap)

CAS 是一种原子操作,通常由处理器提供支持。它涉及三个操作数:内存位置(通常是一个地址)、旧的预期值和新的值。如果内存位置的值与预期值相等,则将新值写入该位置;否则,操作失败。

using System;
using System.Threading;class Program
{static int sharedValue = 0;static void Main(string[] args){// 使用 CAS 算法更新共享变量int expectedValue = 0;int newValue = 1;if (Interlocked.CompareExchange(ref sharedValue, newValue, expectedValue) == expectedValue){Console.WriteLine("Value updated successfully.");}else{Console.WriteLine("Value update failed.");}}
}

在代码中,Interlocked.CompareExchange 方法用于比较并交换操作,它原子性地比较 sharedValue
的值是否等于 expectedValue,如果相等则将 newValue 写入
sharedValue,并返回原来的值;否则不做任何操作。通过这种方式,我们可以实现无锁的并发控制,避免了锁带来的开销和竞争。

CAS 算法通常用于实现无锁的数据结构,例如无锁队列、无锁栈等。虽然 CAS 算法能够提供较好的并发性能,但在某些场景下可能会存在ABA问题等限制,需要特殊处理。

4.2.2 Volatile 关键字

Volatile 关键字用于声明字段是易变的,即可能被多个线程同时访问。它可以确保变量的读取和写入操作都是原子性的,并且不会被编译器或者 CPU 优化掉,从而避免了线程间的数据不一致性问题。

using System;
using System.Threading;class Program
{private static volatile bool _flag = false;static void Main(string[] args){// 启动一个线程不断修改 _flag 的值Thread writerThread = new Thread(WriteFlag);writerThread.Start();// 主线程读取 _flag 的值while (true){if (_flag){Console.WriteLine("Flag is true.");break;}else{Console.WriteLine("Flag is false.");Thread.Sleep(1000);}}}static void WriteFlag(){// 在另一个线程中修改 _flag 的值Thread.Sleep(2000);_flag = true;Console.WriteLine("Flag has been set to true.");}
}

在代码中,使用了 volatile 关键字来声明 _flag 字段,确保了其在多线程环境下的可见性和原子性。主线程不断读取 _flag 的值,而另一个线程在一段时间后将其设置为 true。由于使用了 volatile 关键字,主线程能够正确地读取到 _flag 字段的最新值,从而实现了线程间的正确通信。

4.3 无锁并发编程的优势

  • 减少线程切换开销:无锁并发编程不涉及线程的阻塞和唤醒,可以减少线程切换的开销,提高程序性能。
  • 没有死锁风险:由于无锁并发编程不需要使用锁机制,因此不存在死锁等与锁相关的问题。

4.4 无锁并发编程的局限性

  • 实现复杂度较高:无锁并发编程通常需要仔细设计和实现,因此可能比使用锁机制更复杂。
  • 适用场景有限:无锁并发编程适用于某些特定的场景,例如高并发读操作、轻量级状态同步等。

无锁并发编程是一种重要的并发控制方式,可以提高程序的性能和可伸缩性。但在实际应用中,我们需要根据具体情况选择合适的并发控制方式,以确保程序的正确性和性能。

五、并发集合类

在 C# 中,.NET Framework 提供了许多线程安全的并发集合类,包括 ConcurrentBag、ConcurrentDictionary、ConcurrentQueue 和 ConcurrentStack。本章将介绍这些并发集合类的特点、用途以及示例代码。

5.1 ConcurrentBag

ConcurrentBag 是一个无序的、线程安全的集合类,用于存储对象。它允许多个线程同时添加、移除和遍历元素,适用于需要高度并发性的场景。

using System;
using System.Collections.Concurrent;
using System.Threading.Tasks;class Program
{static void Main(string[] args){ConcurrentBag<int> bag = new ConcurrentBag<int>();// 使用多个线程添加元素到 ConcurrentBagParallel.For(0, 10, i =>{bag.Add(i);Console.WriteLine($"Added {i} to bag.");});// 遍历 ConcurrentBag 中的元素foreach (var item in bag){Console.WriteLine($"Item in bag: {item}");}}
}

5.2 ConcurrentDictionary

ConcurrentDictionary 是一个线程安全的字典集合类,用于存储键值对。它允许多个线程同时对字典进行读取、写入和修改操作,提供了高效的并发性能。

using System;
using System.Collections.Concurrent;class Program
{static void Main(string[] args){ConcurrentDictionary<int, string> dictionary = new ConcurrentDictionary<int, string>();// 使用多个线程添加元素到 ConcurrentDictionaryParallel.For(0, 10, i =>{dictionary.TryAdd(i, i);Console.WriteLine($"{i} Added");});// 读取 ConcurrentDictionary 中的键值对foreach (var kvp in dictionary){Console.WriteLine($"Key: {kvp.Key}, Value: {kvp.Value}");}}
}

5.3 ConcurrentQueue

ConcurrentQueue 是一个线程安全的队列集合类,用于存储对象。它支持多个线程同时对队列进行入队和出队操作,并提供了高效的并发性能。

using System;
using System.Collections.Concurrent;
using System.Threading.Tasks;class Program
{static void Main(string[] args){ConcurrentQueue<int> queue = new ConcurrentQueue<int>();// 使用多个线程入队Parallel.For(0, 10, i =>{queue.Enqueue(i);Console.WriteLine($"Enqueued {i} to queue.");});// 多个线程出队int item;while (queue.TryDequeue(out item)){Console.WriteLine($"Dequeued {item} from queue.");}}
}

5.4 ConcurrentStack

ConcurrentStack 是一个线程安全的栈集合类,用于存储对象。它支持多个线程同时对栈进行入栈和出栈操作,并提供了高效的并发性能。

using System;
using System.Collections.Concurrent;
using System.Threading.Tasks;class Program
{static void Main(string[] args){ConcurrentStack<int> stack = new ConcurrentStack<int>();// 使用多个线程入栈Parallel.For(0, 10, i =>{stack.Push(i);Console.WriteLine($"Pushed {i} to stack.");});// 多个线程出栈int item;while (stack.TryPop(out item)){Console.WriteLine($"Popped {item} from stack.");}}
}

六、经典并发同步问题

以下是几个经典的多线程并发同步问题

6.1 生产者-消费者问题(Producer-Consumer Problem)

生产者线程生成数据并放入共享缓冲区,消费者线程从缓冲区中取出数据进行消费。需要确保在生产者线程生产数据时,消费者线程不会访问空缓冲区,并且在消费者线程消费数据时,生产者线程不会访问满缓冲区。

6.1.1 使用 Monitor 类实现生产者-消费者问题

using System;
using System.Threading;class Program
{static int[] buffer = new int[10];static int count = 0;static object locker = new object();static void Main(string[] args){Thread producerThread = new Thread(Producer);Thread consumerThread = new Thread(Consumer);producerThread.Start();consumerThread.Start();producerThread.Join();consumerThread.Join();}static void Producer(){for (int i = 0; i < 20; i++){lock (locker){while (count == buffer.Length)Monitor.Wait(locker);buffer[count++] = i;Console.WriteLine("Produced: " + i);Monitor.PulseAll(locker);}}}static void Consumer(){for (int i = 0; i < 20; i++){lock (locker){while (count == 0)Monitor.Wait(locker);int consumed = buffer[--count];Console.WriteLine("Consumed: " + consumed);Monitor.PulseAll(locker);}}}
}

6.1.2 使用 Semaphore 类实现生产者-消费者问题

using System;
using System.Threading;class Program
{static int[] buffer = new int[10];static SemaphoreSlim empty = new SemaphoreSlim(10);static SemaphoreSlim full = new SemaphoreSlim(0);static object locker = new object();static void Main(string[] args){Thread producerThread = new Thread(Producer);Thread consumerThread = new Thread(Consumer);producerThread.Start();consumerThread.Start();producerThread.Join();consumerThread.Join();}static void Producer(){for (int i = 0; i < 20; i++){empty.Wait();lock (locker){buffer[i % buffer.Length] = i;Console.WriteLine("Produced: " + i);}full.Release();}}static void Consumer(){for (int i = 0; i < 20; i++){full.Wait();lock (locker){int consumed = buffer[i % buffer.Length];Console.WriteLine("Consumed: " + consumed);}empty.Release();}}
}

6.1.3 使用 BlockingCollection 类实现生产者-消费者问题

using System;
using System.Collections.Concurrent;
using System.Threading;class Program
{static BlockingCollection<int> buffer = new BlockingCollection<int>(10);static void Main(string[] args){Thread producerThread = new Thread(Producer);Thread consumerThread = new Thread(Consumer);producerThread.Start();consumerThread.Start();producerThread.Join();consumerThread.Join();}static void Producer(){for (int i = 0; i < 20; i++){buffer.Add(i);Console.WriteLine("Produced: " + i);}buffer.CompleteAdding();}static void Consumer(){foreach (var item in buffer.GetConsumingEnumerable()){Console.WriteLine("Consumed: " + item);}}
}

这些示例分别使用了 MonitorSemaphoreBlockingCollection 来解决生产者-消费者问题。每个示例都实现了在生产者线程生成数据时,消费者线程不会访问空缓冲区,并且在消费者线程消费数据时,生产者线程不会访问满缓冲区。

6.2 读者-写者问题(Reader-Writer Problem)

多个读者线程可以同时读取共享资源,但写者线程在写入共享资源时需要独占访问。需要确保在有写者写入时,不允许读者读取,以保证数据的一致性。

6.2.1 使用 ReaderWriterLockSlim 类实现读者-写者问题

using System;
using System.Threading;class Program
{static ReaderWriterLockSlim rwLock = new ReaderWriterLockSlim();static int resource = 0;static void Main(string[] args){Thread[] readers = new Thread[5];Thread[] writers = new Thread[2];for (int i = 0; i < 5; i++){readers[i] = new Thread(new ThreadStart(Reader));readers[i].Start();}for (int i = 0; i < 2; i++){writers[i] = new Thread(new ThreadStart(Writer));writers[i].Start();}for (int i = 0; i < 5; i++){readers[i].Join();}for (int i = 0; i < 2; i++){writers[i].Join();}}static void Reader(){while (true){rwLock.EnterReadLock();Console.WriteLine("Reader " + Thread.CurrentThread.ManagedThreadId + " reads: " + resource);rwLock.ExitReadLock();Thread.Sleep(1000);}}static void Writer(){while (true){rwLock.EnterWriteLock();resource++;Console.WriteLine("Writer " + Thread.CurrentThread.ManagedThreadId + " writes: " + resource);rwLock.ExitWriteLock();Thread.Sleep(2000);}}
}

6.2.2 使用 SemaphoreSlim 类实现读者-写者问题

using System;
using System.Threading;class Program
{static SemaphoreSlim readLock = new SemaphoreSlim(1);static SemaphoreSlim writeLock = new SemaphoreSlim(1);static int readersCount = 0;static int resource = 0;static void Main(string[] args){Thread[] readers = new Thread[5];Thread[] writers = new Thread[2];for (int i = 0; i < 5; i++){readers[i] = new Thread(new ThreadStart(Reader));readers[i].Start();}for (int i = 0; i < 2; i++){writers[i] = new Thread(new ThreadStart(Writer));writers[i].Start();}for (int i = 0; i < 5; i++){readers[i].Join();}for (int i = 0; i < 2; i++){writers[i].Join();}}static void Reader(){while (true){readLock.Wait();readersCount++;if (readersCount == 1)writeLock.Wait();readLock.Release();Console.WriteLine("Reader " + Thread.CurrentThread.ManagedThreadId + " reads: " + resource);readLock.Wait();readersCount--;if (readersCount == 0)writeLock.Release();readLock.Release();Thread.Sleep(1000);}}static void Writer(){while (true){writeLock.Wait();resource++;Console.WriteLine("Writer " + Thread.CurrentThread.ManagedThreadId + " writes: " + resource);writeLock.Release();Thread.Sleep(2000);}}
}

6.2.3 使用 Monitor 类实现读者-写者问题

using System;
using System.Threading;class Program
{static object lockObj = new object();static int readersCount = 0;static int resource = 0;static void Main(string[] args){Thread[] readers = new Thread[5];Thread[] writers = new Thread[2];for (int i = 0; i < 5; i++){readers[i] = new Thread(new ThreadStart(Reader));readers[i].Start();}for (int i = 0; i < 2; i++){writers[i] = new Thread(new ThreadStart(Writer));writers[i].Start();}for (int i = 0; i < 5; i++){readers[i].Join();}for (int i = 0; i < 2; i++){writers[i].Join();}}static void Reader(){while (true){lock (lockObj){readersCount++;if (readersCount == 1)Monitor.Enter(lockObj);}Console.WriteLine("Reader " + Thread.CurrentThread.ManagedThreadId + " reads: " + resource);lock (lockObj){readersCount--;if (readersCount == 0)Monitor.Exit(lockObj);}Thread.Sleep(1000);}}static void Writer(){while (true){Monitor.Enter(lockObj);resource++;Console.WriteLine("Writer " + Thread.CurrentThread.ManagedThreadId + " writes: " + resource);Monitor.Exit(lockObj);Thread.Sleep(2000);}}
}

6.3 哲学家就餐问题(Dining Philosophers Problem)

五位哲学家围坐在一张圆桌旁,每位哲学家前面有一只筷子。哲学家思考和进餐,但只有同时拿到两只筷子时才能进餐,而筷子必须是干净的。需要解决资源竞争和死锁的问题。

6.3.1 使用Semaphore实现哲学家就餐问题

using System;
using System.Threading;class Program
{static Semaphore[] sticks = new Semaphore[5];static Semaphore table = new Semaphore(4, 4);static void Main(string[] args){for (int i = 0; i < 5; i++){sticks[i] = new Semaphore(1, 1);}Thread[] philosophers = new Thread[5];for (int i = 0; i < 5; i++){philosophers[i] = new Thread(Philosopher);philosophers[i].Start(i);}for (int i = 0; i < 5; i++){philosophers[i].Join();}}static void Philosopher(object id){int philosopherId = (int)id;while (true){// 思考Console.WriteLine($"Philosopher {philosopherId} is thinking.");// 拿筷子table.WaitOne();sticks[philosopherId].WaitOne();sticks[(philosopherId + 1) % 5].WaitOne();// 吃饭Console.WriteLine($"Philosopher {philosopherId} is eating.");// 放筷子sticks[philosopherId].Release();sticks[(philosopherId + 1) % 5].Release();table.Release();Thread.Sleep(2000);}}
}

6.3.2 使用Mutex实现哲学家就餐问题

using System;
using System.Threading;class Program
{static Mutex[] sticks = new Mutex[5];static void Main(string[] args){for (int i = 0; i < 5; i++){sticks[i] = new Mutex();}Thread[] philosophers = new Thread[5];for (int i = 0; i < 5; i++){philosophers[i] = new Thread(Philosopher);philosophers[i].Start(i);}for (int i = 0; i < 5; i++){philosophers[i].Join();}}static void Philosopher(object id){int philosopherId = (int)id;while (true){// 思考Console.WriteLine($"Philosopher {philosopherId} is thinking.");// 拿筷子sticks[philosopherId].WaitOne();sticks[(philosopherId + 1) % 5].WaitOne();// 吃饭Console.WriteLine($"Philosopher {philosopherId} is eating.");// 放筷子sticks[philosopherId].ReleaseMutex();sticks[(philosopherId + 1) % 5].ReleaseMutex();Thread.Sleep(2000);}}
}

6.3.3 使用Monitor实现哲学家就餐问题

using System;
using System.Threading;class Program
{static object[] sticks = new object[5];static void Main(string[] args){for (int i = 0; i < 5; i++){sticks[i] = new object();}Thread[] philosophers = new Thread[5];for (int i = 0; i < 5; i++){philosophers[i] = new Thread(Philosopher);philosophers[i].Start(i);}for (int i = 0; i < 5; i++){philosophers[i].Join();}}static void Philosopher(object id){int philosopherId = (int)id;while (true){// 思考Console.WriteLine($"Philosopher {philosopherId} is thinking.");lock (sticks[philosopherId]){// 拿左边筷子Monitor.Enter(sticks[philosopherId]);// 拿右边筷子Monitor.Enter(sticks[(philosopherId + 1) % 5]);// 吃饭Console.WriteLine($"Philosopher {philosopherId} is eating.");// 放筷子Monitor.Exit(sticks[philosopherId]);Monitor.Exit(sticks[(philosopherId + 1) % 5]);}Thread.Sleep(2000);}}
}

总结

本文简要探讨了 C# 中的多线程编程技术,重点介绍了锁的基本概念、线程锁的类型、锁的实现方式、无锁并发编程以及 C# 中的并发集合类和经典并发同步问题。通过学习本文,我们可以获得以下几个方面的收获:

  1. 理解多线程编程的基本概念:通过介绍锁的基本概念和原理,可以了解为什么在多线程编程中需要使用锁,以及锁是如何工作的。

  2. 掌握不同类型的线程锁:通过对自旋锁、互斥锁、混合锁和读写锁的介绍,可以了解各种锁的特点、适用场景和实现方式,以便在实际应用中选择合适的锁机制。

  3. 熟悉锁的实现方式:通过对 Monitor、Mutex、Semaphore 和 ReaderWriterLock 的介绍,可以了解不同锁的底层实现原理和使用方法,从而更好地应用于实际开发中。

  4. 了解无锁并发编程:通过介绍无锁算法和无锁并发编程的优势和局限性,可以了解在某些场景下无锁编程可以提供更好的性能和并发能力。

  5. 熟悉 C# 中的并发集合类:通过介绍 ConcurrentBag、ConcurrentDictionary、ConcurrentQueue 和 ConcurrentStack
    等并发集合类,可以了解如何安全地在多线程环境中使用集合类。

  6. 解决经典并发同步问题:通过介绍生产者-消费者问题、读者-写者问题和哲学家就餐问题的解决方案,可以了解如何使用线程锁来解决实际的并发同步问题。

通过本文的学习,可以更加深入地理解并发编程的相关知识,掌握多线程编程的技巧,提高程序的性能和稳定性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/778191.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Python百日进阶-Web开发-Peewee】Day290 - Peewee 的扩展(十)架构迁移(下)/ 映射

文章目录 13.16.3 迁移 APImigrateclass SchemaMigrator(database&#xff09;class PostgresqlMigratorclass SqliteMigratorclass MySQLMigrator 13.17 映射 Reflectiongenerate_modelsprint_modelprint_table_sqlclass Introspectorclassmethod from_database 13.18 数据库网…

进程知识点

引用的文章&#xff1a;操作系统——进程通信&#xff08;IPC&#xff09;_系统ipc-CSDN博客 面试汇总(五)&#xff1a;操作系统常见面试总结(一)&#xff1a;进程与线程的相关知识点 - 知乎 (zhihu.com) 二、进程的定义、组成、组成方式及特征_进程的组成部分必须包含-CSDN博…

江协科技STM32:按键控制LED光敏传感器控制蜂鸣器

按键控制LED LED模块 左上角PA0用上拉输入模式&#xff0c;如果此时引脚悬空&#xff0c;PA0就是高电平&#xff0c;这种方式下&#xff0c;按下按键&#xff0c;引脚为低电平&#xff0c;松下按键&#xff0c;引脚为高电平 右上角PA0&#xff0c;把上拉电阻想象成弹簧 当按键…

大数据手写面试题Scala语言实现大全(持续更新)

在大数据领域,Scala语言因其强大的函数式编程特性和对并发处理的良好支持而成为了开发者们的热门选择。有些面试官,为了考验面试者的基本功,需要让手写一些面试题,以数据结构和算法类的居多。本文将为您提供一些常见的Scala手写面试题及参考答案,帮助您在面试或工作中更好…

C# 命名空间的两种定义哦写法与区别

这两种写法在C#中都是有效的&#xff0c;但是它们代表了不同的语法风格和C#版本特性。 第一种写法&#xff1a; namespace Nebula.PDF; public class PdfDocument {}这是C# 9.0及更高版本中引入的顶级语句&#xff08;top-level statements&#xff09;特性。它允许你直接在文…

c++中的菱形继承

c中的菱形继承 class Animal { public:int m_age; }; class Sheep:public Animal {}; class Tuo:public Animal{}; class SheepTuo :public Sheep , public Tuo{}; SheepTuo st; st.m_age 18; m_age是谁的无法区分 class Animal { public:int m_age; }; class Sheep:virtua…

ES的集群节点发现故障排除指南(3)- end

本文是ES官方文档关于集群节点发现与互联互通的问题排查指南内容&#xff0c;第三部分&#xff0c;结束。 原文参考及相关内容&#xff1a; 英文原文&#xff08;官网&#xff09; 第一部分-&#xff08;1&#xff09; 第二部分-&#xff08;2&#xff09; 节点无法发现或…

李宏毅【生成式AI导论 2024】第5讲 让语言模型彼此合作,把一个人活成一个团队

GPD4,它也有非常强大的能力。但是GPT4如果跟其他的语言模型合作,他们其实可以发挥1加1大于二的力量。 为什么要让模型合作? 那怎么让模型彼此合作呢?有很多不同的方式。一个可能性是假设你现在手边就有一堆语言模型,他们可能有不同的能力使用,他们可能有不同的成本局来…

国外的Java面试题和国内的相比谁更卷

前言 有很多朋友很好奇国外的Java面试题长啥样&#xff0c;今天我们就去找5道国外的面试来和国内的对比一下看看谁难一些&#xff01; 面试题分享 1. Is Java Platform Independent if then how?&#xff08; Java平台是独立的吗&#xff1f;&#xff09; Yes, Java is a…

[数据结构]插入和希尔排序

一、插入排序 插入排序的代码实现虽然没有冒泡排序和选择排序那么简单粗暴&#xff0c;但它的原理应该是最容易理解的了&#xff0c;因为只要打过扑克牌的人都应该能够秒懂。插入排序是一种最简单直观的排序算法&#xff0c;它的工作原理是通过构建有序序列&#xff0c;对于未排…

Protocol Buffers(通常简称为protobuf)是一种由Google开发的用于序列化结构化数据的接口描述语言

Protocol Buffers&#xff08;通常简称为protobuf&#xff09; protobuf是一种由Google开发的用于序列化结构化数据的接口描述语言。它可以用于通信协议、数据存储等领域。以下是protobuf的一些重要特点和概念&#xff1a; IDL&#xff08;接口描述语言&#xff09;&#xff1…

win10微软拼音输入法 - bug - 在PATH变量为空的情况下,无法输入中文

文章目录 win10微软拼音输入法 - bug - 在PATH变量为空的情况下&#xff0c;无法输入中文概述笔记实验前提条件100%可以重现 - 无法使用win10拼音输入法输入中文替代的输入法软件备注END win10微软拼音输入法 - bug - 在PATH变量为空的情况下&#xff0c;无法输入中文 概述 在…

Python数据可视化:用Matplotlib和Seaborn绘制精美图表

Python数据可视化&#xff1a;用Matplotlib和Seaborn绘制精美图表 数据可视化是数据分析中的重要环节&#xff0c;它能够帮助我们以直观的方式理解数据和分析结果。Python中的Matplotlib和Seaborn库是两个非常强大的数据可视化工具&#xff0c;它们提供了丰富的功能来创建各种…

FastAPI+React全栈开发12 搭建FastAPI开发环境

Chapter03 Getting Started with FastAPI 12 Technical requirements FastAPIReact全栈开发12 搭建FastAPI开发环境 For this chapter, you will need the following: Python setupVirtual environmentsCode editor and pluginsTerminalREST clients 对于这一章&#xff0…

Day54:WEB攻防-XSS跨站Cookie盗取表单劫持网络钓鱼溯源分析项目平台框架

目录 XSS跨站-攻击利用-凭据盗取 XSS跨站-攻击利用-数据提交 XSS跨站-攻击利用-flash钓鱼 XSS跨站-攻击利用-溯源综合 知识点&#xff1a; 1、XSS跨站-攻击利用-凭据盗取 2、XSS跨站-攻击利用-数据提交 3、XSS跨站-攻击利用-网络钓鱼 4、XSS跨站-攻击利用-溯源综合 漏洞原理…

OpenCV摄像头和视频处理

OpenCV视频捕获 一、引言 视频捕获是计算机视觉中常见的一项任务&#xff0c;它涉及从视频文件或摄像头实时流中读取帧&#xff0c;并对其进行处理和分析。OpenCV&#xff08;开源计算机视觉库&#xff09;为视频捕获提供了强大的支持&#xff0c;使得这一任务变得简单高效。…

计算机网络链路层

数据链路 链路是从一个节点到相邻节点之间的物理线路&#xff08;有线或无线&#xff09; 数据链路是指把实现协议的软件和硬件加到对应链路上。帧是点对点信道的数据链路层的协议数据单元。 点对点信道 通信的主要步骤&#xff1a; 节点a的数据链路层将网络层交下来的包添…

深度学习pytorch——卷积神经网络(持续更新)

计算机如何解析图片&#xff1f; 在计算机的眼中&#xff0c;一张灰度图片&#xff0c;就是许多个数字组成的二维矩阵&#xff0c;每个数字就是此点的像素值&#xff08;图-1&#xff09;。在存储时&#xff0c;像素值通常位于[0, 255]区间&#xff0c;在深度学习中&#xff0…

重写、重定义(隐藏)、重载区别

1、重载是在同一个作用域中比如在同一个类中、函数名一样参数不同 2、重写&#xff1a; 满足多态的条件&#xff1a;&#xff08;1&#xff09;虚函数前面带有virtual函数名、返回值、参数相同&#xff08;2&#xff09;重写函数体 3、重定义也叫隐藏、不满足重写的就是重定义

素描石膏像的眼睛:传神的关键

素描石膏像的眼睛&#xff1a;如何在素描绘画中画出眼睛的传神&#xff1f;关键要点是什么 在素描石膏像的眼睛时&#xff0c;要画出传神的效果&#xff0c;关键要点包括以下几个方面&#xff1a; 理解眼睛的结构&#xff1a;首先&#xff0c;需要理解眼睛的基本结构&#xff…