Microsoft MVP성태의 닷넷 이야기
.NET Framework: 523. C# 람다(Lambda)에서 변수 캡처 방식 [링크 복사], [링크+제목 복사],
조회: 22900
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 3개 있습니다.)

C# 람다(Lambda)에서 변수 캡처 방식

C# 람다 함수 내에서의 변수 캡처 방식을 한번 살펴볼까요?

예제는 다음의 글에 있는 것으로,

Action 대리자
; https://learn.microsoft.com/en-us/dotnet/api/system.action

가져다 쓰겠습니다.

using System;
using System.Windows.Forms;

public class Name
{
    private string instanceName;

    public Name(string name)
    {
        this.instanceName = name;
    }

    public void DisplayToWindow()
    {
        MessageBox.Show(this.instanceName);
    }
}

public class LambdaExpression
{
    public static void Main()
    {
        Name testName = new Name("Koani");
        Action showMethod = () => testName.DisplayToWindow();
        showMethod();
    }
}

C# 컴파일러는 이런 구문을 만나면 람다 함수내에 캡처되는 변수와 람다 메서드의 코드를 담은 클래스를 컴파일 시에 만들어 둡니다. 가령 다음과 같은 식입니다.

public class [임시클래스]
{
    Name _name;

    public void _f()
    {
        _name.DisplayToWindow(); // showMethod에 넣었던 Lambda 메서드 body
    }
}

그리곤 원래의 소스코드를 다음과 같이 바꿉니다.

public static void Main()
{
    [임시클래스] _var = new [임시클래스]();

    _var._name = new Name("Koani");
    Action showMethod = _var._f;

    showMethod();
}

간단하지요? ^^




이 원칙에 기반해서 C#의 변수 캡처에 대한 주의 사항으로 잘 나오는 예제를 한번 볼까요?

// http://stackoverflow.com/questions/451779/how-to-tell-a-lambda-function-to-capture-a-copy-instead-of-a-reference-in-c

using System;
using System.Collections.Generic;

class Program
{
    static void Main(string[] args)
    {
        List<Action> actions = new List<Action>();

        for (int i = 0; i < 10; ++i)
        {
            actions.Add(() => Console.WriteLine(i));
        }

        foreach (Action a in actions)
        {
            a();
        }

        // 기대하던 출력: 0, 1, 2, 3, 4, 5, 6, 7, 8, 9
        // 실제 출력:    10, 10, 10, 10, 10, 10, 10, 10, 10, 10
    }
}

위의 코드를 작성한 개발자의 기대값과 실제값은 다릅니다. 왜냐하면, C# 컴파일러는 i 값에 대한 변수를 다음과 같이 임시 생성한 클래스의 변수로 대체해 버리기 때문입니다.

// http://stackoverflow.com/questions/451779/how-to-tell-a-lambda-function-to-capture-a-copy-instead-of-a-reference-in-c

using System;
using System.Collections.Generic;

public class [임시클래스]
{
    public int _i;
    public void _f()
    {
        Console.WriteLine(_i);
    }
}

class Program
{
    static void Main(string[] args)
    {
        List<Action> actions = new List<Action>();

        [임시클래스] _var = new [임시클래스]();

        for (_var._i = 0; _var._i < 10; ++_var._i)
        {
            actions.Add(_var._f);
        }

        foreach (Action a in actions)
        {
            a();
        }
    }
}

만약, 개발자가 원래 의도했던 대로 나오게 하고 싶다면 어떻게 해야 할까요? 그럼 다음과 같이 해야 합니다.

using System;
using System.Collections.Generic;
class Program
{
    static void Main(string[] args)
    {
        List<Action> actions = new List<Action>();

        for (int i = 0; i < 10; ++i)
        {
            int v = i;
            actions.Add(() => Console.WriteLine(v));
        }

        foreach (Action a in actions)
        {
            a();
        }
    }
}

이렇게 되면 C# 컴파일러는 i가 아닌 v 변수값을 캡처하기 위해 다음과 같은 식으로 for 루프 내에서 임시클래스를 생성하게 됩니다.

// http://stackoverflow.com/questions/451779/how-to-tell-a-lambda-function-to-capture-a-copy-instead-of-a-reference-in-c

using System;
using System.Collections.Generic;

public class [임시클래스]
{
    public int _v;
    public void _f()
    {
        Console.WriteLine(_i);
    }
}

class Program
{
    static void Main(string[] args)
    {
        List<Action> actions = new List<Action>();

        for (int i = 0; i < 10; ++i)
        {
            [임시클래스] _var = new [임시클래스]();
            _var._v = i;
            actions.Add(_var._f);
        }

        foreach (Action a in actions)
        {
            a();
        }
    }
}

대충 감이 오시나요? ^^ 결국, "마법은 없습니다."

이제 C# 공식 문서의 내용을 보면,

Lambda Expressions (C# Programming Guide)
; https://learn.microsoft.com/en-us/dotnet/csharp/language-reference/operators/lambda-expressions

람다 함수 내에서의 변수 제약이 이해가 됩니다.

  • A variable that is captured will not be garbage-collected until the delegate that references it becomes eligible for garbage collection.
  • Variables introduced within a lambda expression are not visible in the outer method.
  • A lambda expression cannot directly capture a ref or out parameter from an enclosing method.
  • A return statement in a lambda expression does not cause the enclosing method to return.
  • A lambda expression cannot contain a goto statement, break statement, or continue statement that is inside the lambda function if the jump statement’s target is outside the block. It is also an error to have a jump statement outside the lambda function block if the target is inside the block.

참고로, 자바의 변수 캡처 처리 방식과 비교해 보고 싶다면 다음의 글을 참고하세요.

자바 8과 C#의 람다(Lambda) 지원에 대한 비교
; https://www.sysnet.pe.kr/2/0/1685




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 8/31/2023]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 



2015-06-30 05시02분
[spowner] 감사합니다
[guest]
2015-06-30 12시33분
[wafe] 재미있게도 foreach 의 경우에는 capture 룰이 다릅니다 ㅎㅎ c# 5(vs 2012)의 breaking change 중의 하나죠.
[guest]
2015-06-30 12시52분
wafe 님 의견 감사합니다. ^^

Has foreach's use of variables been changed in C# 5?
; http://stackoverflow.com/questions/12112881/has-foreachs-use-of-variables-been-changed-in-c-sharp-5

결국 C# 4에서는 IEnumerator.Current로 받을 변수를 while 문 바깥에 생성했는데, C# 5에서는 안쪽으로 넣었군요. ^^
정성태

... 61  62  63  64  65  66  67  [68]  69  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
11942정성태6/13/201917632개발 환경 구성: 444. 로컬의 Visual Studio Code로 원격 리눅스 머신에 접속해 개발하는 방법 [1]
11941정성태6/13/201910960오류 유형: 546. "message NETSDK1057: You are using a preview version of .NET Core" 빌드 경고 없애는 방법
11940정성태6/13/201910905개발 환경 구성: 443. Visual Studio의 Connection Manager 기능(Remote SSH 관리)을 위한 명령행 도구파일 다운로드1
11939정성태6/13/20199960오류 유형: 545. Managed Debugging Assistant 'FatalExecutionEngineError'
11938정성태6/12/201911594Math: 59. C# - 웨이트 벡터 갱신식을 이용한 퍼셉트론 분류파일 다운로드1
11937정성태6/11/201918064개발 환경 구성: 442. .NET Core 3.0 preview 5를 이용해 Windows Forms/WPF 응용 프로그램 개발 [1]
11936정성태6/10/201911313Math: 58. C# - 최소 자승법의 1차, 2차 수렴 그래프 변화 확인 [2]파일 다운로드1
11935정성태6/9/201912004.NET Framework: 843. C# - PLplot 출력을 파일이 아닌 Window 화면으로 변경
11934정성태6/7/201913186VC++: 133. typedef struct와 타입 전방 선언으로 인한 C2371 오류파일 다운로드1
11933정성태6/7/201913218VC++: 132. enum 정의를 C++11의 enum class로 바꿀 때 유의할 사항파일 다운로드1
11932정성태6/7/201911808오류 유형: 544. C++ - fatal error C1017: invalid integer constant expression파일 다운로드1
11931정성태6/6/201911914개발 환경 구성: 441. C# - CairoSharp/GtkSharp 사용을 위한 프로젝트 구성 방법
11930정성태6/5/201912498.NET Framework: 842. .NET Reflection을 대체할 System.Reflection.Metadata 소개 [1]
11929정성태6/5/201912218.NET Framework: 841. Windows Forms/C# - 클립보드에 RTF 텍스트를 복사 및 확인하는 방법 [1]
11928정성태6/5/201910851오류 유형: 543. PowerShell 확장 설치 시 "Catalog file '[...].cat' is not found in the contents of the module" 오류 발생
11927정성태6/5/201911831스크립트: 15. PowerShell ISE의 스크립트를 복사 후 PPT/Word에 붙여 넣으면 한글이 깨지는 문제 [1]
11926정성태6/4/201913251오류 유형: 542. Visual Studio - pointer to incomplete class type is not allowed
11925정성태6/4/201912058VC++: 131. Visual C++ - uuid 확장 속성과 __uuidof 확장 연산자파일 다운로드1
11924정성태5/30/201913837Math: 57. C# - 해석학적 방법을 이용한 최소 자승법 [1]파일 다운로드1
11923정성태5/30/201913467Math: 56. C# - 그래프 그리기로 알아보는 경사 하강법의 최소/최댓값 구하기파일 다운로드1
11922정성태5/29/201911546.NET Framework: 840. ML.NET 데이터 정규화파일 다운로드1
11921정성태5/28/201916444Math: 55. C# - 다항식을 위한 최소 자승법(Least Squares Method)파일 다운로드1
11920정성태5/28/201910041.NET Framework: 839. C# - PLplot 색상 제어
11919정성태5/27/201913161Math: 54. C# - 최소 자승법의 1차 함수에 대한 매개변수를 단순 for 문으로 구하는 방법 [1]파일 다운로드1
11918정성태5/25/201914357Math: 53. C# - 행렬식을 이용한 최소 자승법(LSM: Least Square Method)파일 다운로드1
11917정성태5/24/201914464Math: 52. MathNet을 이용한 간단한 통계 정보 처리 - 분산/표준편차파일 다운로드1
... 61  62  63  64  65  66  67  [68]  69  70  71  72  73  74  75  ...