푸터 콘텐츠로 바로가기
Iron Academy Logo
C# 배우기
C# 배우기

다른 카테고리

C#의 Yield 소개 - 그게 뭔지, 어떻게 사용하는지, 언제 유용한지

Tim Corey
43분 58초

C#에서 yield 키워드를 처음 접했을 때는 다소 혼란스러울 수 있습니다. 정확히 어떻게 작동하는 건가요? 언제 yield return을 일반적인 return 문 대신 사용해야 할까요? 완벽한 이해를 위해 Tim Corey의 훌륭한 YouTube 튜토리얼 " C#의 Yield 소개 - Yield란 무엇이며, 어떻게 사용하고, 언제 유용한가 "를 바탕으로 자세한 설명을 살펴보겠습니다.

이 가이드에서는 Tim의 비디오에서 특정 부분을 타임스탬프를 통해 참조하여 쉽게 찾아볼 수 있도록 하고, yield를 사용하여 데이터 스트림, 대규모 컬렉션 및 지연 평가를 처리하는 방식을 보여주는 실용적인 예제를 포함합니다.

C#에서 Yield 키워드 소개

Tim은 yield 키워드를 소개하며, 개발자들이 처음 접할 때 혼란스러워할 수 있는 부분을 강조합니다. 그는 yield 문을 사용하면 메서드 실행을 일시 중지하고, 상태를 유지한 다음, 다음 호출 시 중단된 부분부터 다시 시작할 수 있다고 설명합니다. 팀은 특히 대규모 데이터 세트를 다루거나 사용자 지정 반복 로직을 구현할 때 데이터를 효율적으로 처리하려면 수율을 이해하는 것이 매우 중요하다고 강조합니다.

간단한 예제 설정: 클래스 프로그램 및 정적 void main 메서드

집중을 방해하는 요소를 없애기 위해 Tim은 Visual Studio에서 "YieldDemoApp"이라는 간단한 콘솔 애플리케이션을 만듭니다.

Understanding Yield In Csharp 1 related to 간단한 예제 설정: 클래스 프로그램 및 정적 void main 메서드

C#에서 yield는 실제로 무엇을 하는가?

그러자 팀은 이론에 대해 더 깊이 파고든다. 2분 4초에 그는 yield의 동작 방식을 설명합니다. yield 문은 전체 컬렉션을 한 번에 처리하는 대신, 마치 책에 엄지손가락을 끼워두는 것처럼 실행을 일시 중지하고 나중에 다시 시작할 수 있도록 자리를 확보합니다.

이러한 동작은 지연 실행에 매우 중요합니다. 지연 실행에서는 모든 것을 미리 계산하는 대신 필요할 때만 값이 생성됩니다. 팀의 설명은 수익률 계산 방식의 기초를 명확하게 제시합니다.

데모 코드 작성

Program 클래스의 정적 void Main 메서드에서 그는 Console.WriteLine을 사용하여 "앱 시작"과 "앱 종료"와 같은 기본적인 시작 및 종료 메시지를 설정하여 나중에 foreach 루프를 사용할 때 흐름을 명확하게 시각화하는 데 도움을 줍니다.

이 초기 코드 예제는 UI 관련 복잡성을 배제하고 yield 구현에만 초점을 맞추고 있습니다.

PersonModel 클래스 생성

이를 설명하기 위해 Tim은 FirstName과 LastName 속성 및 생성자를 가진 PersonModel 클래스를 만듭니다. PersonModel 객체가 생성될 때, 어떤 사용자가 초기화되었는지 나타내는 메시지가 출력됩니다. 이를 통해 객체가 생성되는 시점과 소비되는 시점을 시각화할 수 있습니다.

Understanding Yield In Csharp 2 related to PersonModel 클래스 생성

이 간단한 코드 생성 단계는 사용자 지정 반복자를 사용하는 데 필요한 기반을 마련합니다.

기존 리스트 반환 방식을 사용한 DataAccess 클래스 구축

5시 6분에 Tim은 IEnumerable을 반환하는 GetPeople 메서드를 가진 DataAccess 클래스로 이동합니다.. 처음에 반환되는 값은 Tim Corey, Sue Storm, Jane Smith라는 세 개의 PersonModel 인스턴스로 채워진 목록입니다.

Understanding Yield In Csharp 3 related to 기존 리스트 반환 방식을 사용한 DataAccess 클래스 구축

이 이터레이터 메서드는 반복이 시작되기 전에 모든 객체를 즉시 메모리에 로드합니다. 이는 Tim이 나중에 yield를 사용하는 것과 비교하는 중요한 점입니다.

Understanding Yield In Csharp 4 related to 기존 리스트 반환 방식을 사용한 DataAccess 클래스 구축

리스트를 이용한 메모리 사용량 시연

foreach 루프를 실행한 후 Tim은 첫 번째 요소가 읽히기도 전에 세 명의 사용자가 모두 생성되었음을 보여줍니다. 이는 대규모 컬렉션이나 대규모 데이터 세트를 다룰 때 발생할 수 있는 잠재적인 문제점, 즉 높은 메모리 사용량을 보여줍니다.

Understanding Yield In Csharp 5 related to 리스트를 이용한 메모리 사용량 시연

팀은 만약 사용자가 천 명이라면, 실제로 필요한 객체가 몇 개 되지 않더라도 천 개의 객체를 생성하게 되어 비효율적인 메모리 할당이 발생한다고 설명합니다.

지연 실행에 대한 수익률로 변경

10시 01분에 Tim은 GetPeople 메서드가 임시 컬렉션(List)을 생성하는 대신 yield return을 사용하도록 수정합니다. 각 yield return 문은 한 번에 하나의 PersonModel을 직접 출력합니다.

이 메서드 내부의 핵심 코드는 지연 평가를 가능하게 하여, foreach 루프에서 다음 요소가 필요할 때만 생성되도록 합니다.

Tim은 또한 해당 메서드의 반환 형식이 IEnumerable이어야 한다고 명확히 밝혔습니다.List를 사용하면 지연 실행의 목적에 어긋나게 됩니다.

디버깅: 컴파일러가 Yield를 위한 코드를 생성하는 방법

팀은 중단점을 사용하여 프로세스를 단계별로 진행합니다. 그는 첫 번째 MoveNext 호출 이전에 시퀀스가 ​​비어 있음을 보여줍니다. foreach 루프가 다음 값을 필요로 할 때만 컴파일러는 이터레이터 블록을 실행하고 yield return num 줄을 실행하여 PersonModel을 초기화하고 반환합니다.

Understanding Yield In Csharp 6 related to 디버깅: 컴파일러가 Yield를 위한 코드를 생성하는 방법

팀은 컴파일러가 일시 중지 및 재개된 실행을 관리하기 위해 내부적으로 특수한 상태 머신을 생성한다는 점을 강조합니다.

수익률 활용의 이점 및 효율성

팀은 수확 효율이 왜 그렇게 높은지 설명합니다.

  • 한 번에 하나의 레코드만 가져올 수 있습니다.

  • 필요한 정수 개수를 제한할 수 있습니다.

  • 이렇게 하면 전체 컬렉션을 메모리에 로드하는 것을 방지할 수 있습니다.

  • 특히 대용량 파일 처리나 데이터 스트림 작업 시 확장성을 향상시킬 수 있습니다.

Tim은 LINQ의 .Take(2)를 사용하여 세 개의 yield return 문이 존재함에도 불구하고 두 개의 객체만 초기화되는 방식을 보여줌으로써 지연 실행의 실제 적용을 강조합니다.

실제 예시: Yield를 이용한 소수 생성기

Tim은 새로운 정적 IEnumerable을 생성합니다.Generators 클래스 내의 GetPrimeNumbers() 메서드. 이 무한 루프에서 Tim은 소수 여부를 확인하기 위해 IsPrimeNumber(int number)라는 도우미 함수를 사용합니다. 숫자가 소수이면 yield return number를 사용하여 해당 숫자를 출력합니다.

이 예시는 무한 수열을 안전하게 처리하는 데 있어 yield가 얼마나 중요한지 보여줍니다.

yield를 사용하지 않으면 무한정 메모리를 소비하여 코드가 충돌할 것입니다. 하지만 수율을 사용하면 지연 실행을 통해 필요에 따라 수치를 생성할 수 있습니다.

Take() 함수를 사용하여 소수 추출하기

그런 다음 Tim은 고정된 개수의 소수만 안전하게 가져오는 방법을 보여줍니다.

var primeNumbers = Generators.GetPrimeNumbers().Take(10000);
var primeNumbers = Generators.GetPrimeNumbers().Take(10000);

static void Main 메서드 안에 작성된 이 코드는 단 10,000개의 소수만 효율적으로 가져옵니다.

수익률 계산 방식은 숫자를 하나씩 순차적으로 생성하기 때문에 메모리 사용량이 낮게 유지됩니다.

사용자 지정 반복: GetEnumerator 및 MoveNext 사용

팀은 한 걸음 더 나아가 반복 작업을 수동으로 제어하는 ​​방법을 설명합니다.

그는 var iterator = primeNumbers.GetEnumerator()를 생성한 다음, int i를 사용하여 for 루프를 실행하고 iterator.MoveNext()를 호출하여 요소를 수동으로 가져옵니다.

이 수동 접근 방식을 통해 사용자 지정 반복이 가능하며, 필요할 때만 다음 값을 요청하고 메서드가 마지막으로 중단된 지점에서 정확히 다시 시작됨을 보여줍니다.

흔히 저지르는 실수: 반환된 컬렉션에 toList()를 사용하는 것

36분 45초에 Tim은 다음과 같이 경고합니다. yield 시퀀스를 .ToList()를 사용하여 List로 변환하면 즉시 전체 평가가 발생합니다.

무한 시퀀스에 대해 .ToList()를 호출하면 앱이 충돌할 위험이 있습니다.

Tim은 yield return이 지연 평가를 위한 것이며, .ToList()를 호출하면 전체 메모리 구체화를 강제하여 이러한 패턴을 깨뜨린다고 강조합니다.

Tim은 LINQ 메서드를 사용할 때 .ToList()를 도입하는 위치에 주의를 기울일 것을 권장합니다.

요약: 수확량 분석이 강력한 도구인 이유

Tim은 yield 키워드가 상태 머신을 유지하는 데 약간의 오버헤드를 발생시키지만, 메모리 사용량 감소와 지연 평가라는 이점 덕분에 다음과 같은 작업을 할 때 강력한 도구가 된다는 점을 강조하며 마무리합니다.

  • 대규모 데이터 세트

  • 대용량 파일

  • 사용자 정의 반복자

  • 무한 수열

  • 데이터 스트림

  • 지연 처리

그는 수익률에 대한 더 깊은 이해를 얻고 흔히 발생하는 함정을 피하기 위해 다양한 프로젝트에서 수익률을 연습해 볼 것을 제안합니다.

팀은 시청자들에게 실제 코드에서 yield를 어떻게 사용했는지 공유해 달라고 요청하며 마무리합니다.

결론

Tim Corey의 영상 을 통해 yield 키워드가 C#에 가져다주는 엄청난 이점을 확인할 수 있었습니다. yield return을 사용하면 사용자 지정 이터레이터를 생성하는 것부터 대규모 컬렉션을 효율적으로 관리하는 것까지 함수 반환을 더욱 스마트하고 메모리 효율적으로 만들 수 있습니다. 변수 number, 변수 point, 변수 reader, 변수 connection 또는 대규모 데이터 세트를 다루든, yield를 숙달하면 C# 코딩 실력을 크게 향상시킬 수 있습니다.

만약 이전에 yield를 사용해 본 적이 없다면, 지금이 간단한 예제를 통해 yield를 연습하고 C# 컴파일러 내부에서 yield return이 어떻게 작동하는지 더 잘 이해할 수 있는 절호의 기회입니다. 더욱 유익한 영상을 보시려면 팀의 공식 유튜브 채널을 방문해 보세요.

Hero Worlddot related to C#의 Yield 소개 - 그게 뭔지, 어떻게 사용하는지, 언제 유용한지
Hero Affiliate related to C#의 Yield 소개 - 그게 뭔지, 어떻게 사용하는지, 언제 유용한지

사랑하는 것을 공유하여 더 많은 수익을 얻으세요

당신은 .NET, C#, Java, Python, 또는 Node.js를 다루는 개발자를 위한 콘텐츠를 만드나요? 당신의 전문성을 추가 수입으로 전환하세요!

아이언 서포트 팀

저희는 주 5일, 24시간 온라인으로 운영합니다.
채팅
이메일
전화해