Microsoft MVP성태의 닷넷 이야기
.NET Framework: 523. C# 람다(Lambda)에서 변수 캡처 방식 [링크 복사], [링크+제목 복사],
조회: 31487
글쓴 사람
정성태 (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에서는 안쪽으로 넣었군요. ^^
정성태

... 31  32  33  34  35  36  37  38  39  [40]  41  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
12942정성태1/27/202216380.NET Framework: 1141. XmlSerializer와 Dictionary 타입파일 다운로드1
12941정성태1/26/202217597오류 유형: 790. AKS/k8s - pod 상태가 Pending으로 지속되는 경우
12940정성태1/26/202214035오류 유형: 789. AKS에서 hpa에 따른 autoscale 기능이 동작하지 않는다면?
12939정성태1/25/202215164.NET Framework: 1140. C# - ffmpeg(FFmpeg.AutoGen)를 이용해 MP3 오디오 파일 인코딩/디코딩하는 예제파일 다운로드1
12938정성태1/24/202218377개발 환경 구성: 633. Docker Desktop + k8s 환경에서 local 이미지를 사용하는 방법
12937정성태1/24/202216085.NET Framework: 1139. C# - ffmpeg(FFmpeg.AutoGen)를 이용해 오디오(mp2) 인코딩하는 예제(encode_audio.c) [2]파일 다운로드1
12936정성태1/22/202215558.NET Framework: 1138. C# - ffmpeg(FFmpeg.AutoGen)를 이용해 멀티미디어 파일의 메타데이터를 보여주는 예제(metadata.c)파일 다운로드1
12935정성태1/22/202216261.NET Framework: 1137. ffmpeg의 파일 해시 예제(ffhash.c)를 C#으로 포팅파일 다운로드1
12934정성태1/22/202215691오류 유형: 788. Warning C6262 Function uses '65564' bytes of stack: exceeds /analyze:stacksize '16384'. Consider moving some data to heap. [2]
12933정성태1/21/202216105.NET Framework: 1136. C# - ffmpeg(FFmpeg.AutoGen)를 이용해 MP2 오디오 파일 디코딩 예제(decode_audio.c)파일 다운로드1
12932정성태1/20/202217254.NET Framework: 1135. C# - ffmpeg(FFmpeg.AutoGen)로 하드웨어 가속기를 이용한 비디오 디코딩 예제(hw_decode.c) [2]파일 다운로드1
12931정성태1/20/202213737개발 환경 구성: 632. ASP.NET Core 프로젝트를 AKS/k8s에 올리는 과정
12930정성태1/19/202214968개발 환경 구성: 631. AKS/k8s의 Volume에 파일 복사하는 방법
12929정성태1/19/202215011개발 환경 구성: 630. AKS/k8s의 Pod에 Volume 연결하는 방법
12928정성태1/18/202214757개발 환경 구성: 629. AKS/Kubernetes에서 호스팅 중인 pod에 shell(/bin/bash)로 진입하는 방법
12927정성태1/18/202215432개발 환경 구성: 628. AKS 환경에 응용 프로그램 배포 방법
12926정성태1/17/202215197오류 유형: 787. AKS - pod 배포 시 ErrImagePull/ImagePullBackOff 오류
12925정성태1/17/202215949개발 환경 구성: 627. AKS의 준비 단계 - ACR(Azure Container Registry)에 docker 이미지 배포
12924정성태1/15/202217387.NET Framework: 1134. C# - ffmpeg(FFmpeg.AutoGen)를 이용한 비디오 디코딩 예제(decode_video.c) [2]파일 다운로드1
12923정성태1/15/202216172개발 환경 구성: 626. ffmpeg.exe를 사용해 비디오 파일을 MPEG1 포맷으로 변경하는 방법
12922정성태1/14/202214977개발 환경 구성: 625. AKS - Azure Kubernetes Service 생성 및 SLO/SLA 변경 방법
12921정성태1/14/202212648개발 환경 구성: 624. Docker Desktop에서 별도 서버에 설치한 docker registry에 이미지 올리는 방법
12920정성태1/14/202213971오류 유형: 786. Camtasia - An error occurred with the camera: Failed to Add Video Sampler.
12919정성태1/13/202213531Windows: 199. Host Network Service (HNS)에 의해서 점유되는 포트
12918정성태1/13/202214016Linux: 47. WSL - shell script에서 설정한 환경 변수가 스크립트 실행 후 반영되지 않는 문제
12917정성태1/12/202213237오류 유형: 785. C# - The type or namespace name '...' could not be found (are you missing a using directive or an assembly reference?)
... 31  32  33  34  35  36  37  38  39  [40]  41  42  43  44  45  ...