Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)
(시리즈 글이 2개 있습니다.)
.NET Framework: 561. null 처리된 객체가 왜 GC에 의해 수집되지 않을까요?
; https://www.sysnet.pe.kr/2/0/10920

닷넷: 2154. C# - 네이티브 자원을 포함한 관리 개체(예: 스레드)의 GC 정리
; https://www.sysnet.pe.kr/2/0/13435




null 처리된 객체가 왜 GC에 의해 수집되지 않을까요?

다음과 같은 질문이 있군요.

C# WeakReference이 CPU 플랫폼 설정마다 결과가 틀리게 나옵니다.
; https://www.sysnet.pe.kr/3/0/4702

문제를 정리해 보면, 다음과 같이 코딩을 한 경우 (제 환경에서) Debug 빌드로 하면 Target이 살아 있는 걸로 나옵니다.

// .NET 4.5.2 + Debug 빌드 테스트 (x86/x64)
public class Man
{
    public string Name
    {
        get;
        private set;
    }

    public Man(string name)
    {
        Name = name;
    }
}

class Program
{
    static void Main(string[] args)
    {
        TestWeakReference();

        Console.ReadKey();
    }

    private static void TestWeakReference()
    {
        Man m = new Man("A");
        WeakReference refMan = new WeakReference(m);

        m = null;

        GC.Collect();

        Console.WriteLine("약한 참조 원본: {0}", (m == null ? "null" : m.Name));
        Console.WriteLine("약한 참조 참조: {0}", (refMan.Target == null ? "null" : (refMan.Target as Man).Name));
    }
}

// 출력 결과
약한 참조 원본: null
약한 참조 참조: A

코드 상으로 보면 분명 "m = null"로 되었고 GC.Collect가 수행되었으니 당연히 WeakReference의 Target이 null이 나와야 합니다. 그런데, 왜 여전히 살아있을까요?

사실, 제가 답변에도 썼지만 마이크로소프트 입장에서는 WeakReference의 Target이 정확히 언제 null로 된다거나, 심지어 힙 객체가 '정해진 시점'에 해제된다고 명시하고 있진 않습니다. 즉, 그 시점을 정확히 예측해서 어떤 부가적인 코드를 작성하는 것은 바람직하지 않습니다.

그래도, 저런 결과가 나오면 좀 이상하긴 할 텐데요. 어디... 그 원인을 한번 밝혀 볼까요? ^^

일단, 우리가 배운 바로는 GC가 되려면 또 다른 참조 객체, 스택 및 레지스터(CPU Register)에 그 참조가 없어야 한다는 것입니다. 그렇다면, 저 경우에 분명히 그 세 가지 중의 하나는 어디에선가 참조값이 있어야 합니다.

그중에서 우선 "또 다른 참조 객체"는 후보가 아닙니다. 코드상에 WeakReference가 'm'의 참조값을 가지고는 있지만 GC에게 Weak 참조는 GC 대상에서 고려하지 않기 때문에 무시해도 됩니다. 나머지 후보라면 스택 및 레지스터가 있는데요. 이를 살펴보기 위해 다음과 같이 소스 코드에 Console.ReadKey()를 추가하고,

private static void TestWeakReference()
{
    Man m = new Man("A");

    WeakReference refMan = new WeakReference(m);

    Console.ReadKey();

    m = null;

    GC.Collect();

    Console.WriteLine("약한 참조 원본: {0}", (m == null ? "null" : m.Name));
    Console.WriteLine("약한 참조 참조: {0}", (refMan.Target == null ? "null" : (refMan.Target as Man).Name));

    Console.ReadKey();
}

실행한 후 Visual Studio를 이용해 "Attach to Process..."로 연결한 다음 저 상태에서의 기계어 코드를 확인해 봅니다. 그럼, 대략 다음과 같은 화면을 볼 수 있습니다.

gc_weak_ref_1.png

여기서, 생성된 m에 대한 인스턴스를 보관하고 있는 곳은 2군데입니다.

하나는 가장 상단의 "call 00007FFDDCB24C10"로 표현된 Man 객체의 생성자 호출에 이은 "mov qword ptr [rbp-40h], rax"로 "[rbp-40h]" 스택에 보관되어 있고,

00007FFD7D4D0510 E8 FB 46 65 5F       call        00007FFDDCB24C10  
00007FFD7D4D0515 48 89 45 C0          mov         qword ptr [rbp-40h],rax  

또 하나는 [rbp-40h]의 값을 다시 [rbp-30h]로 이동해서 보관해 놓은 것입니다.

00007FFD7D4D052F 48 8B 4D C0          mov         rcx,qword ptr [rbp-40h]  
00007FFD7D4D0533 48 89 4D D0          mov         qword ptr [rbp-30h],rcx  

반면 하단에서 "m = null" 할 때는 [rbp-30h] 값만 0으로 대입하고 있습니다.

        m = null;
00007FFD7D4D0569 33 C0                xor         eax,eax  
00007FFD7D4D056B 48 89 45 D0          mov         qword ptr [rbp-30h],rax  

실제로 [rbp-40h]의 값과 [rbp-30h]의 값이 같다는 것을 메모리 창을 통해서 확인할 수 있습니다. 레지스터 창을 통해 RBP == 0000006B335FEE10로 나오니까, 각각 다음과 같이 계산되고,

[rbp-40h] == 0x0000006B335FEDE0
[rbp-30h] == 0x0000006B335FEDD0

이에 대한 메모리 값을 확인해 보면 "e8 45 a4 40 5a 02 00 00"로 동일하게 나옵니다.

gc_weak_ref_2.png

따라서, GC는 [rbp-30h]의 참조가 없어졌다 해도 [rbp-40h]의 값으로 인해 해당 객체를 제거하지 못했던 것입니다.

어디... 그럼 실험으로 증명해 볼까요? ^^

해당 스택의 값을 비주얼 스튜디오의 메모리 창을 통해 임의로 '00 00 00 00 00 00 00 00'으로 초기화할 수 있습니다.

gc_weak_ref_3.png

이렇게 해준 다음 실행을 계속해 보면 다음과 같이 성공적으로 참조 해제가 된 것을 확인할 수 있습니다.

약한 참조 원본: null
약한 참조 참조: null

그러니까, 결국 JIT 컴파일러가 생성해 낸 기계어 코드에서 Debug 빌드인 경우 스택에 두 번 보관해 놓은 값 때문에 저런 문제(?)가 발생한 것입니다.




이 때문에 Release 빌드로 하면 이런 문제가 발생하지 않습니다. 릴리스 빌드 시 JIT 컴파일러는 최적화된 코드를 생성하기 때문에 쓸데없는 스택 낭비를 유발하는 코드를 생성하지 않기 때문입니다.

이 외에도 코드를 다음과 같이 바꿔주면 Debug 빌드 시에도 약한 참조가 끊기게 됩니다.

// .NET 4.5.2 + Debug/Release 빌드 테스트 (x86/x64)
class Program
{
    static void Main(string[] args)
    {
        WeakReference refMan = TestWeakReference();
        GC.Collect();

        Console.WriteLine("약한 참조 참조: {0}", (refMan.Target == null ? "null" : (refMan.Target as Man).Name));
        Console.ReadKey();
    }

    private static WeakReference TestWeakReference()
    {
        Man m = new Man("A");

        WeakReference refMan = new WeakReference(m);

        m = null;

        Console.WriteLine("약한 참조 원본: {0}", (m == null ? "null" : m.Name));

        return refMan;
    }
}

// 출력 결과
약한 참조 원본: null
약한 참조 참조: null

이유를 아시겠죠? ^^ 이런 경우 디버그 빌드로 해도 스택에 2중으로 보관되어 있던 값이 메서드가 리턴하면서 스택이 해제되기 때문에 CLR의 GC 구성 요소가 검사해야 할 영역에서 제외되기 때문입니다.

(첨부한 코드는 이 글의 예제를 포함합니다.)




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 11/7/2023]

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

비밀번호

댓글 작성자
 



2016-06-09 08시29분
[kernel0] 궁금한게 있습니다. m = null; 부근쯤에 breakpoint 잡고 start debugging 으로 작업하지 않고, attach to process 로 붙이는 이유는 뭔가요? debug heap 사용 같은 것을 피하려고 하는 건가요?
[guest]
2016-06-09 09시13분
@kernel0 가능한 debugger로 영향받는 것을 없애기 위해 그렇게 한 것입니다. 일례로, 본문에서 rbp-xxh로 나오는데 디버거로 연결해 두면 rbp+xxh와 같은 식으로 변화가 발생합니다.
정성태
2016-06-10 05시01분
[kernel0] 답변 감사드립니다 :) native c++ 같이 디버거가 있건 없건 disassembly가 같은 줄 알았습니다.
[guest]
2016-08-21 09시03분
[tera] 감사합니다! 스크랩을..
[guest]
2016-08-21 09시08분
[tera] 원문링크 포함해서 스크랩해갈게요! 궁금했던 내용이었는데 정말 감사합니다.
[guest]
2019-04-15 04시54분
Suppress JIT Optimization On Module Load (Managed Only)
; https://learn.microsoft.com/en-us/visualstudio/debugger/jit-optimization-and-debugging
정성태

1  2  3  4  5  6  7  8  9  10  11  12  13  [14]  15  ...
NoWriterDateCnt.TitleFile(s)
13280정성태3/9/20234606개발 환경 구성: 670. WSL 2에서 호스팅 중인 TCP 서버를 외부에서 접근하는 방법
13279정성태3/9/20234125오류 유형: 851. 파이썬 ModuleNotFoundError: No module named '_cffi_backend'
13278정성태3/8/20234157개발 환경 구성: 669. WSL 2의 (init이 아닌) systemd 지원 [1]
13277정성태3/6/20234812개발 환경 구성: 668. 코드 사인용 인증서 신청 및 적용 방법(예: Digicert)
13276정성태3/5/20234477.NET Framework: 2102. C# 11 - ref struct/ref field를 위해 새롭게 도입된 scoped 예약어
13275정성태3/3/20234793.NET Framework: 2101. C# 11의 ref 필드 설명
13274정성태3/2/20234356.NET Framework: 2100. C# - ref 필드로 ref struct 타입을 허용하지 않는 이유
13273정성태2/28/20234086.NET Framework: 2099. C# - 관리 포인터로서의 ref 예약어 의미
13272정성태2/27/20234357오류 유형: 850. SSMS - mdf 파일을 Attach 시킬 때 Operating system error 5: "5(Access is denied.)" 에러
13271정성태2/25/20234274오류 유형: 849. Sql Server Configuration Manager가 시작 메뉴에 없는 경우
13270정성태2/24/20233872.NET Framework: 2098. dotnet build에 /p 옵션을 적용 시 유의점
13269정성태2/23/20234437스크립트: 46. 파이썬 - uvicorn의 콘솔 출력을 UDP로 전송
13268정성태2/22/20234988개발 환경 구성: 667. WSL 2 내부에서 열고 있는 UDP 서버를 호스트 측에서 접속하는 방법
13267정성태2/21/20234899.NET Framework: 2097. C# - 비동기 소켓 사용 시 메모리 해제가 finalizer 단계에서 발생하는 사례파일 다운로드1
13266정성태2/20/20234506오류 유형: 848. .NET Core/5+ - Process terminated. Couldn't find a valid ICU package installed on the system
13265정성태2/18/20234432.NET Framework: 2096. .NET Core/5+ - PublishSingleFile 유형에 대한 runtimeconfig.json 설정
13264정성태2/17/20235933스크립트: 45. 파이썬 - uvicorn 사용자 정의 Logger 작성
13263정성태2/16/20234087개발 환경 구성: 666. 최신 버전의 ilasm.exe/ildasm.exe 사용하는 방법
13262정성태2/15/20235160디버깅 기술: 191. dnSpy를 이용한 (소스 코드가 없는) 닷넷 응용 프로그램 디버깅 방법 [1]
13261정성태2/15/20234422Windows: 224. Visual Studio - 영문 폰트가 Fullwidth Latin Character로 바뀌는 문제
13260정성태2/14/20234231오류 유형: 847. ilasm.exe 컴파일 오류 - error : syntax error at token '-' in ... -inf
13259정성태2/14/20234379.NET Framework: 2095. C# - .NET5부터 도입된 CollectionsMarshal
13258정성태2/13/20234265오류 유형: 846. .NET Framework 4.8 Developer Pack 설치 실패 - 0x81f40001
13257정성태2/13/20234353.NET Framework: 2094. C# - Job에 Process 포함하는 방법 [1]파일 다운로드1
13256정성태2/10/20235187개발 환경 구성: 665. WSL 2의 네트워크 통신 방법 - 두 번째 이야기
13255정성태2/10/20234518오류 유형: 845. gihub - windows2022 이미지에서 .NET Framework 4.5.2 미만의 프로젝트에 대한 빌드 오류
1  2  3  4  5  6  7  8  9  10  11  12  13  [14]  15  ...