Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 3개 있습니다.)

C# - 구조체의 크기가 16바이트가 넘어가면 힙에 할당된다?

아래와 같은 질문에서,

시작하세요 C# 9.0,  225페이지 구조체 관련 질문드립니다.
; https://www.sysnet.pe.kr/3/0/5489

구조체의 크기가 16바이트를 넘어가면 힙에 할당된다고... 하는데... 아니... 도대체 그런 근거 없는 사실을 퍼뜨리고 있는 사람이 누굴까요? ^^;

구조체는 크기가 어떻게 되었든 무조건 스택에 할당됩니다. 그래도, 말로만 하면 재미가 없으니 눈으로 확인할 수 있도록 테스트를 해봐야겠지요. ^^ 방법은, 다음과 같이 매우 간단합니다.

using System;
using System.Threading;

public struct Size64
{
    public long l1;
    public long l2;
    public long l3;
    public long l4;

    public long l5;
    public long l6;
    public long l7;
    public long l8;
}

class Program
{
    static void Main(string[] args)
    {
        Thread t = new Thread(threadFunc);
        t.IsBackground = true;
        t.Start();

        while (true)
        {
            int n = GC.CollectionCount(0) + GC.CollectionCount(1) + GC.CollectionCount(2);
            Console.WriteLine(n);

            Thread.Sleep(1000);
        }
    }

    private static void threadFunc()
    {
        while (true)
        {
            Size64 variable = new Size64();
        }
    }
}

/* 출력 결과
0
0
0
0
0
...[생략]...
*/

보다시피 threadFunc는 지속적으로 Size64 구조체를 new 시키고 있는데요, 따라서 16바이트가 넘어가는 구조체가 정말 힙에 할당된다면 Main에서 GC.CollectionCount를 구하는 코드에서 n 값은 계속 증가해야 합니다. 물론, 실행해 보면 n 값은 언제까지고 0으로만 나옵니다.

반대로 Size64를 구조체가 아닌 class로 정의하면,

public class Size64
{
    public long l1;
    public long l2;
    public long l3;
    public long l4;

    public long l5;
    public long l6;
    public long l7;
    public long l8;
}

/* 출력 결과
0
1984
4420
6959
9454
11650
14186
...[생략]...
*/

이제는 GC.CollectionCount가 반환하는 n 값이 빠르게 증가하는 것을 확인할 수 있습니다.




혹시나 그럼, 어느 크기에선가는 heap으로 할당되지 않을까요? 이에 대한 답이 예전에 설명한 예제에 있습니다.

CER(Constrained Execution Region)이란?
; https://www.sysnet.pe.kr/2/0/11868#large_sized_struct

fixed 예약어를 사용하면 struct의 크기를 1MB 크기까지 간단하게 늘릴 수 있는데요,

// 스택 크기 기본값 - 32비트 1MB, 64비트 4MB
// 따라서 32비트로 빌드한 경우

unsafe struct Big
{
    public fixed byte Bytes[1_048_576]; // 일반적인 스택 메모리의 크기가 1MB이므로 StackOverflowException 발생
}

따라서, 위의 구조체를 생성하면 (stack에 할당을 시도하므로) StackOverflowException 예외가 발생합니다.

// 32비트로 빌드한 경우

Big big = new Big(); // StackOverflowException 발생

// 또는,

Big big;  // 구조체의 특성상 new를 하지 않아도 할당하므로 StackOverflowException 발생

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




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

[연관 글]






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

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

비밀번호

댓글 작성자
 



2021-04-30 12시17분
궁금해서 찾아 보니 아래의 글에서 16바이트 이야기가 나오고 있군요. ^^

C#/.NET 구조체(struct)와 클래스(class) 차이
; https://hijuworld.tistory.com/43

Unity C# 클래스와 구조체의 차이점
; https://dhy948.tistory.com/13

게다가 또 한 가지 잘못된 정보가 있는데요,

"2. 구조체안에 클래스 타입을 필드로 가질 경우이다."

구조체 안에 클래스 타입을 필드로 가진다고 해서 "구조체" 자체가 힙에 할당되는 것은 아닙니다. 단지, 클래스 타입의 필드는 그 인스턴스만 힙에 있고 그 인스턴스를 가리키는 포인터 값은 구조체 내에 존재하며 따라서 스택에 구조체가 유지된다는 사실에는 변함이 없습니다.
정성태
2022-11-05 06시21분
[yopeule] 감사합니다! 저도 같은 부분들에 의문이 들어 찾아왔는데요.
"구조체안에 클래스 타입의 필드가 있으면 힙에 할당된다"는 말은, 아마도 거꾸로 작성된 게 아닐까 합니다.
클래스 타입 안에 구조체가 있으면 그 구조체는 클래스에 인라인되는데, 즉, 힙에 할당되게 되죠. 그걸 거꾸로 쓰신게 아닐까? 하는 생각이 듭니다.

정리하자면, 클래스 안에 구조체 타입 필드가 있으면 해당 필드는 클래스와 함께 힙에 할당된다는 내용이 와전된 것 같습니다.
[guest]

... 46  47  48  49  50  51  52  53  54  55  56  57  58  59  [60]  ...
NoWriterDateCnt.TitleFile(s)
12140정성태2/10/202010087.NET Framework: 887. C# - ASP.NET 웹 응용 프로그램의 출력 가로채기파일 다운로드1
12139정성태2/9/202011454.NET Framework: 886. C# - Console 응용 프로그램에서 UI 스레드 구현 방법
12138정성태2/9/202014227.NET Framework: 885. C# - 닷넷 응용 프로그램에서 SQLite 사용 [6]파일 다운로드1
12137정성태2/9/20209363오류 유형: 592. [AhnLab] 경고 - 디버거 실행을 탐지했습니다.
12136정성태2/6/20209808Windows: 168. Windows + S(또는 Q)로 뜨는 작업 표시줄의 검색 바가 동작하지 않는 경우
12135정성태2/6/202013432개발 환경 구성: 468. Nuget 패키지의 로컬 보관 폴더를 옮기는 방법 [2]
12134정성태2/5/202013542.NET Framework: 884. eBEST XingAPI의 C# 래퍼 버전 - XingAPINet Nuget 패키지 [5]파일 다운로드1
12133정성태2/5/202010785디버깅 기술: 161. Windbg 환경에서 확인해 본 .NET 메서드 JIT 컴파일 전과 후 - 두 번째 이야기
12132정성태1/28/202012368.NET Framework: 883. C#으로 구현하는 Win32 API 후킹(예: Sleep 호출 가로채기)파일 다운로드1
12131정성태1/27/202012428개발 환경 구성: 467. LocaleEmulator를 이용해 유니코드를 지원하지 않는(한글이 깨지는) 프로그램을 실행하는 방법 [1]
12130정성태1/26/20209917VS.NET IDE: 142. Visual Studio에서 windbg의 "Open Executable..."처럼 EXE를 직접 열어 디버깅을 시작하는 방법
12129정성태1/26/202015443.NET Framework: 882. C# - 키움 Open API+ 사용 시 Registry 등록 없이 KHOpenAPI.ocx 사용하는 방법 [3]
12128정성태1/26/202010260오류 유형: 591. The code execution cannot proceed because mfc100.dll was not found. Reinstalling the program may fix this problem.
12127정성태1/25/202010113.NET Framework: 881. C# DLL에서 제공하는 Win32 export 함수의 내부 동작 방식(VT Fix up Table)파일 다운로드1
12126정성태1/25/202010953.NET Framework: 880. C# - PE 파일로부터 IMAGE_COR20_HEADER 및 VTableFixups 테이블 분석파일 다운로드1
12125정성태1/24/20208792VS.NET IDE: 141. IDE0019 - Use pattern matching
12124정성태1/23/202010620VS.NET IDE: 140. IDE1006 - Naming rule violation: These words must begin with upper case characters: ...
12123정성태1/23/202012123웹: 39. Google Analytics - gtag 함수를 이용해 페이지 URL 수정 및 별도의 이벤트 생성 방법 [2]
12122정성태1/20/20209074.NET Framework: 879. C/C++의 UNREFERENCED_PARAMETER 매크로를 C#에서 우회하는 방법(IDE0060 - Remove unused parameter '...')파일 다운로드1
12121정성태1/20/20209660VS.NET IDE: 139. Visual Studio - Error List: "Could not find schema information for the ..."파일 다운로드1
12120정성태1/19/202011099.NET Framework: 878. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 - 네 번째 이야기(IL 코드로 직접 구현)파일 다운로드1
12119정성태1/17/202011130디버깅 기술: 160. Windbg 확장 DLL 만들기 (3) - C#으로 만드는 방법
12118정성태1/17/202011805개발 환경 구성: 466. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 - 세 번째 이야기 [1]
12117정성태1/15/202010790디버깅 기술: 159. C# - 디버깅 중인 프로세스를 강제로 다른 디버거에서 연결하는 방법파일 다운로드1
12116정성태1/15/202011271디버깅 기술: 158. Visual Studio로 디버깅 시 sos.dll 확장 명령어를 (비롯한 windbg의 다양한 기능을) 수행하는 방법
12115정성태1/14/202011038디버깅 기술: 157. C# - PEB.ProcessHeap을 이용해 디버깅 중인지 확인하는 방법파일 다운로드1
... 46  47  48  49  50  51  52  53  54  55  56  57  58  59  [60]  ...