멀티스레드 C#의 foreach 루프 및 클로저 이해
C#의 멀티스레딩은 foreach
루프 및 클로저를 사용할 때 복잡성을 초래할 수 있습니다. foreach
루프 내의 클로저가 루프의 반복 변수를 참조할 때 문제가 발생합니다.
다음 예를 고려해보세요.
<code class="language-csharp">var threads = new List<Thread>(); foreach (Foo f in ListOfFoo) { Thread thread = new Thread(() => f.DoSomething()); threads.Add(thread); thread.Start(); }</code>
여기서 각 스레드의 클로저는 단일 f
변수에 대한 참조를 캡처합니다. 루프가 진행됨에 따라 f
의 값이 변경되어 예측할 수 없는 결과가 발생합니다. 스레드는 예상치 못한 Foo
개체
스레드 안전성 확보
이를 방지하려면 각 반복마다 새 지역 변수를 생성하세요.
<code class="language-csharp">var threads = new List<Thread>(); foreach (Foo f in ListOfFoo) { Foo f2 = f; // Create a copy for each thread Thread thread = new Thread(() => f2.DoSomething()); threads.Add(thread); thread.Start(); }</code>
이제 각 스레드에는 고유한 독립적인 참조(f2
)가 있어 DoSomething()
가 올바른 Foo
인스턴스에서 호출되도록 보장합니다.
C# 5 이상
C# 5 이상에서는 동작이 약간 변경되었습니다. 컴파일러는 foreach
변수가 루프 범위 외부에 나타나더라도 루프 범위 내에서 효과적으로 선언된 것으로 처리합니다. 따라서 위의 두 코드 조각은 모두 C# 5 및 후속 버전에서 스레드로부터 안전합니다. 그러나 명시적 사본은 명확성과 이전 버전과의 호환성을 위해 여전히 좋은 습관입니다.
위 내용은 다중 스레드 C# 코드에서 Foreach 루프 및 클로저에 특별한 처리가 필요한 이유는 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!