람다 표현식에서 캡처한 변수: "foreach 루프 변수" 미스터리 풀기
목록 생성을 목표로 하는 아래의 복잡한 코드를 고려해보세요. 다양한 데이터 유형을 환영하는 메소드:
public class MyClass { public delegate string PrintHelloType(string greeting); public void Execute() { Type[] types = new Type[]{typeof(string), typeof(float), typeof(int)}; List<PrintHelloType> helloMethods = new List<PrintHelloType>(); foreach (var type in types) { var sayHello = new PrintHelloType(greeting => SayGreetingToType(type, greeting)); helloMethods.Add(sayHello); } foreach (var helloMethod in helloMethods) { Console.WriteLine(helloMethod("Hi")); } } public string SayGreetingToType(Type type, string greetingText) { return greetingText + " " + type.Name; } }
그러나 이 코드를 실행하면 예상치 못한 결과가 발생합니다. 의도한 "Hi String", "Hi" 대신 "Hi Int32"가 세 번 인쇄됩니다. 싱글', '안녕 Int32'. 여기서 무슨 일이 벌어지고 있는 걸까요?
클로저와 캡처된 변수의 미묘함
이 동작을 이해하는 열쇠는 클로저와 캡처된 변수의 개념에 있습니다. 람다 식이 생성되면 주변 범위에서 변수를 캡처합니다. 이 경우 람다 표현식은 foreach 루프에서 유형 변수를 캡처합니다.
캡처된 루프 변수: 함정
결정적으로 캡처된 변수는 다음의 값이 아닙니다. 유형이 아니라 변수 자체입니다. 루프가 진행됨에 따라 유형 변수가 변경되지만 람다 표현식은 여전히 캡처된 동일한 변수를 참조합니다.
문제 수정
의도한 동작을 달성하려면 의도한 값을 캡처하려면 루프 내에서 새 변수를 생성해야 합니다. 코드를 수정하는 방법은 다음과 같습니다.
foreach (var type in types) { var newType = type; var sayHello = new PrintHelloType(greeting => SayGreetingToType(newType, greeting)); helloMethods.Add(sayHello); }
type 대신 newType을 캡처하면 이제 람다 표현식이 의도한 값에 대한 상수 참조를 갖게 되며 예상되는 출력을 얻습니다.
결론
람다 표현식을 사용할 때는 클로저와 캡처된 변수의 복잡성을 이해하는 것이 필수적입니다. 이러한 메커니즘을 염두에 두면 예상치 못한 동작을 방지하고 코드가 의도한 대로 작동하도록 할 수 있습니다.
위 내용은 foreach 루프의 모든 람다 식이 개별 값 대신 동일한 변수를 캡처하는 이유는 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!