Microsoft MVP성태의 닷넷 이야기
디버깅 기술: 19. C++/CLI - F11 디버깅 시의 변수 초기화 [링크 복사], [링크+제목 복사],
조회: 22080
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일


C++/CLI - F11 디버깅 시의 변수 초기화


개발하다 보면, 심심치 않게 만나게 되는 당혹스러운 사건 중에 하나가 바로 디버깅 시와 그렇지 않은 경우에 발생하는 실행 결과의 차이인데요. 사실, 디버그 빌드와 릴리스 빌드 자체의 차이점 같은 경우는 정규 학습으로는 배울 수가 없지요. 하물며, 디버그 빌드에 "디버거"의 재량에 따라 달라지는 실행 차이는 마이크로소프트의 블로그 웹 사이트에서도 잘 찾아볼 수 없습니다.

일례로, WaitForSingleObject/MsgWaitForMultipleObjects 같은 동기화 함수를 사용할 때 Queue에 스레드가 대기하게 되는데, 일반적인 상황에서는 Queue에 쌓인 순서대로 스레드가 깨어나게 되는데, 디버거 상에서 Break Point에 의한 디버깅이 개입되었을 때는 Queue에 쌓인 스레드가 재정렬된다고 합니다. 만약, 누군가 FIFO대로 실행되고 있는지 테스트하려고 한다면 그냥 실행하면 잘 되는데, VS.NET IDE에서 디버깅하면 안되는 현상을 만나게 될 것입니다.

제가 최근에 겪은 현상은 그와는 반대였습니다.

즉, 디버깅상에서는 정상적으로 실행이 되고, 그냥 실행하면 안되는 경우였습니다. ^^;

상황은 간단합니다. .NET 프로젝트에서 C++/CLI 함수를 부른 것인데, 가령 C++/CLI에서 정의된 타입은 아래와 같습니다.

namespace CppLib 
{

    public ref class Class1
    {
    public:

        int IntValue;
        double DoubleValue;
        System::Guid ^GuidValue;

        void MyTestMethod()
        {
            
            int intValue;   // 초기화 안 했음.
            double doubleValue;
            GUID guid;


            this->IntValue = intValue;
            this->DoubleValue = doubleValue;
            this->GuidValue = gcnew Guid(guid.Data1, guid.Data2, guid.Data3,
                guid.Data4[0], guid.Data4[1],guid.Data4[2],guid.Data4[3],guid.Data4[4],
                guid.Data4[5],guid.Data4[6],guid.Data4[7]);
        }
    };
}

닷넷 프로젝트에서는 위의 함수를 다음과 같이 불러서 테스트해 봅니다.

namespace WindowsFormsApplication1
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            CppLib.Class1 cl1 = new CppLib.Class1();
            cl1.MyTestMethod();
            Debug.WriteLine(string.Format("{0}, {1}, {2}", cl1.IntValue, cl1.DoubleValue, cl1.GuidValue));
        }
    }
}

자, 이제 그냥 실행을 해보면, "Debug.WriteLine" 코드로 인해서 다음과 같이 출력이 됩니다.

0, 0, 00adeabf-e308-067b-b45f-2802e4992802

원래, C++에서는 지역변수는 쓰레기 값으로 채워져 있지만 C++/CLI는 ... 그래도 닷넷이라고 지역변수가 기본적으로 초기화되어 있는 것을 확인할 수 있습니다. 하지만, GUID 변수는 보시는 것처럼 "00adeabf-e308-067b-b45f-2802e4992802"와 같이 쓰레기 값이 들어가 있지요.

여기서, 하나 더 테스트를 해서, "Form1.Form1_Load" 메서드의 "cl1.MyTestMethod();" 코드에 Break Point를 잡고, "F11(Step Into)" 기능을 이용해서 MyTestMethod 안으로 디버깅을 들어가봅니다. 그런 경우에 실행해 보면, 다음과 같은 Debug.WriteLine 결과를 볼 수 있습니다.

0, 0, 00000000-0000-0000-0000-000000000000

모든 변수가 초기화 되었습니다. ^^

제가 만든 모듈은, Guid 값이 Guid_NULL인 상황에서만 정상적으로 동작할 수 있는 것이었는데, 위의 문제 때문에 디버깅으로 실행해서 들어가면 정상적으로 실행이 되다가도, 그냥 실행하면 오류가 발생하는 현상을 겪게 된 것이었습니다.


첨부된 파일은 위의 테스트 예제 솔루션입니다.



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







[최초 등록일: ]
[최종 수정일: 3/25/2023]

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)
13893정성태2/27/20252225Linux: 115. eBPF (bpf2go) - ARRAY / HASH map 기본 사용법
13892정성태2/24/20252983닷넷: 2325. C# - PowerShell과 연동하는 방법파일 다운로드1
13891정성태2/23/20252500닷넷: 2324. C# - 프로세스의 성능 카운터용 인스턴스 이름을 구하는 방법파일 다운로드1
13890정성태2/21/20252329닷넷: 2323. C# - 프로세스 메모리 중 Private Working Set 크기를 구하는 방법(Win32 API)파일 다운로드1
13889정성태2/20/20253054닷넷: 2322. C# - 프로세스 메모리 중 Private Working Set 크기를 구하는 방법(성능 카운터, WMI) [1]파일 다운로드1
13888정성태2/17/20252498닷넷: 2321. Blazor에서 발생할 수 있는 async void 메서드의 부작용
13887정성태2/17/20253074닷넷: 2320. Blazor의 razor 페이지에서 code-behind 파일로 코드를 분리 및 DI 사용법
13886정성태2/15/20252574VS.NET IDE: 196. Visual Studio - Code-behind처럼 cs 파일을 그룹핑하는 방법
13885정성태2/14/20253236닷넷: 2319. ASP.NET Core Web API / Razor 페이지에서 발생할 수 있는 async void 메서드의 부작용
13884정성태2/13/20253522닷넷: 2318. C# - (async Task가 아닌) async void 사용 시의 부작용파일 다운로드1
13883정성태2/12/20253266닷넷: 2317. C# - Memory Mapped I/O를 이용한 PCI Configuration Space 정보 열람파일 다운로드1
13882정성태2/10/20252581스크립트: 70. 파이썬 - oracledb 패키지 연동 시 Thin / Thick 모드
13881정성태2/7/20252832닷넷: 2316. C# - Port I/O를 이용한 PCI Configuration Space 정보 열람파일 다운로드1
13880정성태2/5/20253172오류 유형: 947. sshd - Failed to start OpenSSH server daemon.
13879정성태2/5/20253407오류 유형: 946. Ubuntu - N: Updating from such a repository can't be done securely, and is therefore disabled by default.
13878정성태2/3/20253198오류 유형: 945. Windows - 최대 절전 모드 시 DRIVER_POWER_STATE_FAILURE 발생 (pacer.sys)
13877정성태1/25/20253249닷넷: 2315. C# - PCI 장치 열거 (레지스트리, SetupAPI)파일 다운로드1
13876정성태1/25/20253708닷넷: 2314. C# - ProcessStartInfo 타입의 Arguments와 ArgumentList파일 다운로드1
13875정성태1/24/20253137스크립트: 69. 파이썬 - multiprocessing 패키지의 spawn 모드로 동작하는 uvicorn의 workers
13874정성태1/24/20253558스크립트: 68. 파이썬 - multiprocessing Pool의 기본 프로세스 시작 모드(spawn, fork)
13873정성태1/23/20252983디버깅 기술: 217. WinDbg - PCI 장치 열거파일 다운로드1
13872정성태1/23/20252885오류 유형: 944. WinDbg - 원격 커널 디버깅이 연결은 되지만 Break (Ctrl + Break) 키를 눌러도 멈추지 않는 현상
13871정성태1/22/20253292Windows: 278. Windows - 윈도우를 다른 모니터 화면으로 이동시키는 단축키 (Window + Shift + 화살표)
13870정성태1/18/20253731개발 환경 구성: 741. WinDbg - 네트워크 커널 디버깅이 가능한 NIC 카드 지원 확대
13869정성태1/18/20253456개발 환경 구성: 740. WinDbg - _NT_SYMBOL_PATH 환경 변수에 설정한 경로로 심벌 파일을 다운로드하지 않는 경우
13868정성태1/17/20253112Windows: 277. Hyper-V - Windows 11 VM의 Enhanced Session 모드로 로그인을 할 수 없는 문제
1  [2]  3  4  5  6  7  8  9  10  11  12  13  14  15  ...