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]

... 31  32  33  [34]  35  36  37  38  39  40  41  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
13093정성태7/5/202215160.NET Framework: 2029. C# - HttpWebRequest로 localhost 접속 시 2초 이상 지연
13092정성태7/3/202216798.NET Framework: 2028. C# - HttpWebRequest의 POST 동작 방식파일 다운로드1
13091정성태7/3/202215385.NET Framework: 2027. C# - IPv4, IPv6를 모두 지원하는 서버 소켓 생성 방법
13090정성태6/29/202214635오류 유형: 815. PyPI에 업로드한 패키지가 반영이 안 되는 경우
13089정성태6/28/202215354개발 환경 구성: 646. HOSTS 파일 변경 시 Edge 브라우저에 반영하는 방법
13088정성태6/27/202213704개발 환경 구성: 645. "Developer Command Prompt for VS 2022" 명령행 환경의 폰트를 바꾸는 방법
13087정성태6/23/202217691스크립트: 41. 파이썬 - FastAPI / uvicorn 호스팅 환경에서 asyncio 사용하는 방법 [1]
13086정성태6/22/202217155.NET Framework: 2026. C# 11 - 문자열 보간 개선 2가지파일 다운로드1
13085정성태6/22/202216846.NET Framework: 2025. C# 11 - 원시 문자열 리터럴(raw string literals)파일 다운로드1
13084정성태6/21/202215708개발 환경 구성: 644. Windows - 파이썬 2.7을 msi 설치 없이 구성하는 방법
13083정성태6/20/202216243.NET Framework: 2024. .NET 7에 도입된 GC의 메모리 해제에 대한 segment와 region의 차이점 [2]
13082정성태6/19/202215275.NET Framework: 2023. C# - Process의 I/O 사용량을 보여주는 GetProcessIoCounters Win32 API파일 다운로드1
13081정성태6/17/202214434.NET Framework: 2022. C# - .NET 7 Preview 5 신규 기능 - System.IO.Stream ReadExactly / ReadAtLeast파일 다운로드1
13080정성태6/17/202215491개발 환경 구성: 643. Visual Studio 2022 17.2 버전에서 C# 11 또는 .NET 7.0 preview 적용
13079정성태6/17/202212918오류 유형: 814. 파이썬 - Error: The file/path provided (...) does not appear to exist
13078정성태6/16/202216208.NET Framework: 2021. WPF - UI Thread와 Render Thread파일 다운로드1
13077정성태6/15/202217068스크립트: 40. 파이썬 - PostgreSQL 환경 구성
13075정성태6/15/202214042Linux: 50. Linux - apt와 apt-get의 차이 [2]
13074정성태6/13/202214948.NET Framework: 2020. C# - NTFS 파일에 사용자 정의 속성값 추가하는 방법파일 다운로드1
13073정성태6/12/202215136Windows: 207. Windows Server 2022에 도입된 WSL 2
13072정성태6/10/202215348Linux: 49. Linux - ls 명령어로 출력되는 디렉터리 색상 변경 방법
13071정성태6/9/202215982스크립트: 39. Python에서 cx_Oracle 환경 구성
13070정성태6/8/202216619오류 유형: 813. Windows 11에서 입력 포커스가 바뀌는 문제 [1]
13069정성태5/26/202218668.NET Framework: 2019. C# - .NET에서 제공하는 3가지 Timer 비교 [2]
13068정성태5/24/202217315.NET Framework: 2018. C# - 일정 크기를 할당하는 동안 GC를 (가능한) 멈추는 방법 [1]파일 다운로드1
13067정성태5/23/202215358Windows: 206. Outlook - 1년 이상 지난 메일이 기본적으로 안 보이는 문제
... 31  32  33  [34]  35  36  37  38  39  40  41  42  43  44  45  ...