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)
13426정성태10/13/20233287스크립트: 59. 파이썬 - 비동기 호출 함수(run_until_complete, run_in_executor, create_task, run_in_threadpool)
13425정성태10/11/20233105닷넷: 2149. C# - PLinq의 Partitioner<T>를 이용한 사용자 정의 분할파일 다운로드1
13423정성태10/6/20233084스크립트: 58. 파이썬 - async/await 기본 사용법
13422정성태10/5/20233224닷넷: 2148. C# - async 유무에 따른 awaitable 메서드의 병렬 및 예외 처리
13421정성태10/4/20233254닷넷: 2147. C# - 비동기 메서드의 async 예약어 유무에 따른 차이
13420정성태9/26/20235356스크립트: 57. 파이썬 - UnboundLocalError: cannot access local variable '...' where it is not associated with a value
13419정성태9/25/20233108스크립트: 56. 파이썬 - RuntimeError: dictionary changed size during iteration
13418정성태9/25/20233790닷넷: 2146. C# - ConcurrentDictionary 자료 구조의 동기화 방식
13417정성태9/19/20233354닷넷: 2145. C# - 제네릭의 형식 매개변수에 속한 (매개변수를 가진) 생성자를 호출하는 방법
13416정성태9/19/20233161오류 유형: 877. redis-py - MISCONF Redis is configured to save RDB snapshots, ...
13415정성태9/18/20233643닷넷: 2144. C# 12 - 컬렉션 식(Collection Expressions)
13414정성태9/16/20233409디버깅 기술: 193. Windbg - ThreadStatic 필드 값을 조사하는 방법
13413정성태9/14/20233603닷넷: 2143. C# - 시스템 Time Zone 변경 시 이벤트 알림을 받는 방법
13412정성태9/14/20236877닷넷: 2142. C# 12 - 인라인 배열(Inline Arrays) [1]
13411정성태9/12/20233387Windows: 252. 권한 상승 전/후 따로 관리되는 공유 네트워크 드라이브 정보
13410정성태9/11/20234894닷넷: 2141. C# 12 - Interceptor (컴파일 시에 메서드 호출 재작성) [1]
13409정성태9/8/20233751닷넷: 2140. C# - Win32 API를 이용한 모니터 전원 끄기
13408정성태9/5/20233739Windows: 251. 임의로 만든 EXE 파일을 포함한 ZIP 파일의 압축을 해제할 때 Windows Defender에 의해 삭제되는 경우
13407정성태9/4/20233495닷넷: 2139. C# - ParallelEnumerable을 이용한 IEnumerable에 대한 병렬 처리
13406정성태9/4/20233450VS.NET IDE: 186. Visual Studio Community 버전의 라이선스
13405정성태9/3/20233869닷넷: 2138. C# - async 메서드 호출 원칙
13404정성태8/29/20233398오류 유형: 876. Windows - 키보드의 등호(=, Equals sign) 키가 눌리지 않는 경우
13403정성태8/21/20233230오류 유형: 875. The following signatures couldn't be verified because the public key is not available: NO_PUBKEY EB3E94ADBE1229CF
13402정성태8/20/20233297닷넷: 2137. ILSpy의 nuget 라이브러리 버전 - ICSharpCode.Decompiler
13401정성태8/19/20233535닷넷: 2136. .NET 5+ 환경에서 P/Invoke의 성능을 높이기 위한 SuppressGCTransition 특성 [1]
13400정성태8/10/20233376오류 유형: 874. 파이썬 - pymssql을 윈도우 환경에서 설치 불가
1  2  3  4  5  6  7  [8]  9  10  11  12  13  14  15  ...