lock 문 - 공유 리소스에 대한 단독 액세스 확인
lock
문은 지정된 개체에 대한 상호 배제 잠금을 획득하여 명령문 블록을 실행한 다음, 잠금을 해제합니다. 잠금이 유지되는 동안 잠금을 보유하는 스레드는 잠금을 다시 획득하고 해제할 수 있습니다. 다른 스레드는 잠금을 획득할 수 없도록 차단되며 잠금이 해제될 때까지 대기합니다. lock
문은 한 번에 최대 하나의 스레드만 본문을 실행하도록 합니다.
lock
문이 형식입니다.
lock (x)
{
// Your code...
}
여기서 x
는 참조 형식의 식입니다. 정확히 다음과 같은 경우
object __lockObj = x;
bool __lockWasTaken = false;
try
{
System.Threading.Monitor.Enter(__lockObj, ref __lockWasTaken);
// Your code...
}
finally
{
if (__lockWasTaken) System.Threading.Monitor.Exit(__lockObj);
}
코드에서 try-finally
문을 사용하므로 lock
문의 본문 내에서 예외가 throw되더라도 잠금이 해제됩니다.
lock
문의 본문에는 await
식을 사용할 수 없습니다.
지침
공유 리소스에 대한 스레드 액세스를 동기화하는 경우 전용 개체 인스턴스(예: private readonly object balanceLock = new object();
) 또는 코드의 관련 없는 파트에서 잠금 개체로 사용되지 않을 가능성이 있는 다른 인스턴스를 잠급니다. 교착 상태 또는 잠금 경합이 발생할 수 있으므로 다른 공유 리소스에 대해 동일한 잠금 개체 인스턴스를 사용하지 마세요. 특히 다음 인스턴스를 잠금 개체로 사용하지 않도록 합니다.
this
(호출자가 잠금으로 사용할 수 있음).- Type 인스턴스는 typeof 연산자 또는 리플렉션에서 가져올 수 있습니다.
- 문자열 인스턴스(문자열 리터럴 포함)(인터닝될 수 있음).
잠금 경합을 줄이기 위해 최대한 짧은 시간 동안 잠금을 유지하세요.
예시
다음 예제에서는 전용 balanceLock
인스턴스에 잠금을 설정하여 해당 개인 balance
필드에 대한 액세스를 동기화하는 Account
클래스를 정의합니다. 동일한 인스턴스를 잠금에 사용하면 Debit
또는 Credit
메서드를 동시에 호출하려는 두 스레드에 의해 balance
필드가 동시에 업데이트되지 않습니다.
using System;
using System.Threading.Tasks;
public class Account
{
private readonly object balanceLock = new object();
private decimal balance;
public Account(decimal initialBalance) => balance = initialBalance;
public decimal Debit(decimal amount)
{
if (amount < 0)
{
throw new ArgumentOutOfRangeException(nameof(amount), "The debit amount cannot be negative.");
}
decimal appliedAmount = 0;
lock (balanceLock)
{
if (balance >= amount)
{
balance -= amount;
appliedAmount = amount;
}
}
return appliedAmount;
}
public void Credit(decimal amount)
{
if (amount < 0)
{
throw new ArgumentOutOfRangeException(nameof(amount), "The credit amount cannot be negative.");
}
lock (balanceLock)
{
balance += amount;
}
}
public decimal GetBalance()
{
lock (balanceLock)
{
return balance;
}
}
}
class AccountTest
{
static async Task Main()
{
var account = new Account(1000);
var tasks = new Task[100];
for (int i = 0; i < tasks.Length; i++)
{
tasks[i] = Task.Run(() => Update(account));
}
await Task.WhenAll(tasks);
Console.WriteLine($"Account's balance is {account.GetBalance()}");
// Output:
// Account's balance is 2000
}
static void Update(Account account)
{
decimal[] amounts = [0, 2, -3, 6, -2, -1, 8, -5, 11, -6];
foreach (var amount in amounts)
{
if (amount >= 0)
{
account.Credit(amount);
}
else
{
account.Debit(Math.Abs(amount));
}
}
}
}
C# 언어 사양
자세한 내용은 C# 언어 사양의 lock 문 섹션을 참조하세요.
참고 항목
.NET
피드백
https://aka.ms/ContentUserFeedback
출시 예정: 2024년 내내 콘텐츠에 대한 피드백 메커니즘으로 GitHub 문제를 단계적으로 폐지하고 이를 새로운 피드백 시스템으로 바꿀 예정입니다. 자세한 내용은 다음을 참조하세요.다음에 대한 사용자 의견 제출 및 보기