Using Locks in C#
Understanding Locks in C#
**Locks** in C# are used to prevent **race conditions** when multiple threads access shared resources simultaneously. The `lock` statement, `Monitor`, `Mutex`, `Semaphore`, and `ReaderWriterLock` are commonly used synchronization mechanisms.
Key Features of Locks
- Ensures **thread safety** when accessing shared data.
- Prevents **race conditions** by allowing only one thread at a time.
- Different types of locks are available based on requirements (`lock`, `Monitor`, `Mutex`, `Semaphore`).
- May cause **deadlocks** if not used properly.
Using the Lock Statement
The `lock` statement provides a simple way to **synchronize access** to shared resources.
Example: Using `lock` for Thread Synchronization
using System;
using System.Threading;
class Program
{
static readonly object _lock = new object();
static int counter = 0;
static void IncrementCounter()
{
for (int i = 0; i < 5; i++)
{
lock (_lock)
{
counter++;
Console.WriteLine($"Thread {Thread.CurrentThread.ManagedThreadId}: Counter = {counter}");
}
Thread.Sleep(500);
}
}
static void Main()
{
Thread thread1 = new Thread(IncrementCounter);
Thread thread2 = new Thread(IncrementCounter);
thread1.Start();
thread2.Start();
thread1.Join();
thread2.Join();
Console.WriteLine("Final Counter Value: " + counter);
}
}
// Output (Ensures safe counter increment):
// Thread 1: Counter = 1
// Thread 2: Counter = 2
// Thread 1: Counter = 3
// Thread 2: Counter = 4
// Final Counter Value: 10
The **lock (_lock)** statement ensures that only **one thread** can modify `counter` at a time.
Using the Monitor Class
The `Monitor` class provides **more control** over locking than the `lock` statement.
Example: Using Monitor for Thread Synchronization
using System;
using System.Threading;
class Program
{
static readonly object _lock = new object();
static int counter = 0;
static void IncrementCounter()
{
for (int i = 0; i < 5; i++)
{
bool lockTaken = false;
try
{
Monitor.Enter(_lock, ref lockTaken);
counter++;
Console.WriteLine($"Thread {Thread.CurrentThread.ManagedThreadId}: Counter = {counter}");
}
finally
{
if (lockTaken) Monitor.Exit(_lock);
}
Thread.Sleep(500);
}
}
static void Main()
{
Thread thread1 = new Thread(IncrementCounter);
Thread thread2 = new Thread(IncrementCounter);
thread1.Start();
thread2.Start();
thread1.Join();
thread2.Join();
Console.WriteLine("Final Counter Value: " + counter);
}
}
// Output (Ensures safe counter increment):
// Thread 1: Counter = 1
// Thread 2: Counter = 2
// Final Counter Value: 10
`Monitor.Enter` provides **fine-grained control** over locking, allowing explicit unlocking in a `finally` block.
Using Mutex for Process Synchronization
`Mutex` allows **cross-process synchronization**, ensuring only one process can access a resource at a time.
Example: Using Mutex for Synchronization
using System;
using System.Threading;
class Program
{
static Mutex mutex = new Mutex();
static void AccessResource()
{
mutex.WaitOne(); // Acquire the lock
Console.WriteLine($"Thread {Thread.CurrentThread.ManagedThreadId} is accessing the resource.");
Thread.Sleep(2000);
Console.WriteLine($"Thread {Thread.CurrentThread.ManagedThreadId} released the resource.");
mutex.ReleaseMutex(); // Release the lock
}
static void Main()
{
Thread thread1 = new Thread(AccessResource);
Thread thread2 = new Thread(AccessResource);
thread1.Start();
thread2.Start();
thread1.Join();
thread2.Join();
}
}
// Output (Ensures only one thread accesses the resource at a time):
// Thread 1 is accessing the resource.
// (Waits for 2 seconds)
// Thread 1 released the resource.
// Thread 2 is accessing the resource.
// Thread 2 released the resource.
**Mutex.WaitOne()** blocks other threads until the lock is released using **Mutex.ReleaseMutex()**.
Best Practices for Using Locks
- Use **lock** for **simple, single-threaded synchronization**.
- Use **Monitor** for **fine-grained locking control** with `try-finally`.
- Use **Mutex** when **synchronizing across multiple processes**.
- Minimize **lock duration** to reduce **performance overhead**.
- Avoid **nested locks**, as they can lead to **deadlocks**.