Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

C# - Local '...' or its members cannot have their address taken and be used inside an anonymous method or lambda expression

예를 들어, 다음과 같이 코드를 작성하면,

using System;

internal class Program
{
    static unsafe void Main(string[] args)
    {
        MyStruct* ptr = null;
        MyStruct** ptr2 = &ptr;

        Action action = () =>
        {
            if (ptr == null)
            {
                Console.Write("TEST");
            }
        };

        t.Start();
    }
}

public struct MyStruct
{
    public int x;
}

컴파일 오류가 발생합니다.

Error CS1686 Local 'ptr' or its members cannot have their address taken and be used inside an anonymous method or lambda expression

위에서 "MyStruct** ptr2 = &ptr;" 코드가 없거나, ptr 포인터 사용을 익명 메서드 또는 람다식에서 사용하지 않으면 오류가 발생하지 않습니다.

그 이유는? C# 컴파일러는 익명 메서드에서 ptr 변수를 캡처하기 위해 익명 클래스와 함께 그것의 필드에 ptr의 현재 값을 저장해 둡니다. 하지만, 그 ptr의 포인터를 다시 2중 포인터에 대입함으로써 나중에 ptr의 값이 바뀔 것을 예측할 수 있지만 정작 캡처된 변수는 Main 메서드가 가진 ptr 변수의 값을 복사만 한 것이므로 ptr2 이중 포인터의 동작과는 무관하게 됩니다. 즉, 사용자의 의도와는 달리 action 내부의 메서드에서 (캡처된) ptr 변수는 그냥 영원히 null에 불과합니다.

이 상황에서 컴파일 오류를 해결하려면, 단순히 ptr 변수가 아닌 ptr2 변수를 넘기면 됩니다.

MyStruct* ptr = null;
MyStruct** ptr2 = &ptr;

Action action = () =>
{
    if (*ptr2 == null)
    {
        Console.Write("TEST");
    }
};

하지만, 오류의 원인을 설정했던 컴파일러의 걱정을 무시하면 안 됩니다. 즉, 저런 코드에서 개발자는 반드시 익명 메서드 또는 람다식의 실행이 저 코드를 포함한 메서드가 실행 중인 동안에만 유효하게끔 주의를 기울여야 합니다.

즉, 아래와 같은 식으로 지역 변수가 포함된 메서드 내에서 실행을 완료하는 경우에만 안전한 코드가 됩니다.

using System;

internal class Program
{
    static unsafe void Main(string[] args)
    {
        MyStruct data = new MyStruct();
        data.x = 500;
        MyStruct* ptr = &data;
        MyStruct** ptr2 = &ptr;

        Action action = () =>
        {
            if (*ptr2 != null)
            {
                Console.Write($"TEST: {(*ptr2)->x}"); // 출력 결과 TEST: 500
            }
        };

        action();
    }
}

public struct MyStruct
{
    public int x;
}

그렇지 않고, 스택이 파괴될 수 있는 상황에서는,

static unsafe void Main(string[] args)
{
    Action action = GetAction();
    Span<byte> ptr = stackalloc byte[100];
    for (int i = 0; i < ptr.Length; i ++) // 이전 스택 파괴
    {
        ptr[i] = 0x70;
    }
    action();
}

private static unsafe Action GetAction()
{
    MyStruct data = new MyStruct();
    data.x = 500;
    MyStruct* ptr = null;
    MyStruct** ptr2 = &ptr;

    Action action = () =>
    {
        if (*ptr2 != null)
        {
            IntPtr p = new IntPtr(*ptr2);
            Console.WriteLine($"p == {p.ToString("x")}");
            Console.Write($"TEST: {(*ptr2)->x}");
        }
    };

    *ptr2 = &data;

    return action;
}

다음과 같은 식의 출력 결과를 얻게 됩니다.

p == 7070707070707070
Fatal error. System.AccessViolationException: Attempted to read or write protected memory. This is often an indication that other memory is corrupt.
   at Program+<>c__DisplayClass1_0.<GetAction>b__0()
   at Program.Main(System.String[])





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







[최초 등록일: ]
[최종 수정일: 3/21/2022]

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

비밀번호

댓글 작성자
 




1  2  3  4  5  6  7  8  9  10  11  [12]  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13329정성태4/24/20233706Windows: 246. Win32 C/C++ - 직접 띄운 대화창 템플릿을 위한 Modal 메시지 루프 생성파일 다운로드1
13328정성태4/19/20233379VS.NET IDE: 184. Visual Studio - Fine Code Coverage에서 동작하지 않는 Fake/Shim 테스트
13327정성태4/19/20233793VS.NET IDE: 183. C# - .NET Core/5+ 환경에서 Fakes를 이용한 단위 테스트 방법
13326정성태4/18/20235204.NET Framework: 2109. C# - 닷넷 응용 프로그램에서 SQLite 사용 (System.Data.SQLite) [1]파일 다운로드1
13325정성태4/18/20234500스크립트: 48. 파이썬 - PostgreSQL의 with 문을 사용한 경우 연결 개체 누수
13324정성태4/17/20234318.NET Framework: 2108. C# - Octave의 "save -binary ..."로 생성한 바이너리 파일 분석파일 다운로드1
13323정성태4/16/20234259개발 환경 구성: 677. Octave에서 Excel read/write를 위한 io 패키지 설치
13322정성태4/15/20235036VS.NET IDE: 182. Visual Studio - 32비트로만 빌드된 ActiveX와 작업해야 한다면?
13321정성태4/14/20233826개발 환경 구성: 676. WSL/Linux Octave - Python 스크립트 연동
13320정성태4/13/20233804개발 환경 구성: 675. Windows Octave 8.1.0 - Python 스크립트 연동
13319정성태4/12/20234290개발 환경 구성: 674. WSL 2 환경에서 GNU Octave 설치
13318정성태4/11/20234112개발 환경 구성: 673. JetBrains IDE에서 "Squash Commits..." 메뉴가 비활성화된 경우
13317정성태4/11/20234225오류 유형: 855. WSL 2 Ubuntu 20.04 - error: cannot communicate with server: Post http://localhost/v2/snaps/...
13316정성태4/10/20233552오류 유형: 854. docker-compose 시 "json.decoder.JSONDecodeError: Expecting value: line 1 column 1 (char 0)" 오류 발생
13315정성태4/10/20233750Windows: 245. Win32 - 시간 만료를 갖는 컨텍스트 메뉴와 윈도우 메시지의 영역별 정의파일 다운로드1
13314정성태4/9/20233828개발 환경 구성: 672. DosBox를 이용한 Turbo C, Windows 3.1 설치
13313정성태4/9/20233911개발 환경 구성: 671. Hyper-V VM에 Turbo C 2.0 설치 [2]
13312정성태4/8/20233932Windows: 244. Win32 - 시간 만료를 갖는 MessageBox 대화창 구현 (개선된 버전)파일 다운로드1
13311정성태4/7/20234441C/C++: 163. Visual Studio 2022 - DirectShow 예제 컴파일(WAV Dest)
13310정성태4/6/20234020C/C++: 162. Visual Studio - /NODEFAULTLIB 옵션 설정 후 수동으로 추가해야 할 library
13309정성태4/5/20234206.NET Framework: 2107. .NET 6+ FileStream의 구조 변화
13308정성태4/4/20234090스크립트: 47. 파이썬의 time.time() 실숫값을 GoLang / C#에서 사용하는 방법
13307정성태4/4/20233863.NET Framework: 2106. C# - .NET Core/5+ 환경의 Windows Forms 응용 프로그램에서 HINSTANCE 구하는 방법
13306정성태4/3/20233664Windows: 243. Win32 - 윈도우(cbWndExtra) 및 윈도우 클래스(cbClsExtra) 저장소 사용 방법
13305정성태4/1/20234032Windows: 242. Win32 - 시간 만료를 갖는 MessageBox 대화창 구현 (쉬운 버전)파일 다운로드1
13304정성태3/31/20234387VS.NET IDE: 181. Visual Studio - C/C++ 프로젝트에 application manifest 적용하는 방법
1  2  3  4  5  6  7  8  9  10  11  [12]  13  14  15  ...