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

닷넷 메타데이터에 struct/class(값/참조 형식)의 구분이 있을까요?

메타데이터 테이블로부터 타입에 대한 정보를 조회할 수 있는 메서드가 있습니다.

IMetaDataImport::GetTypeDefProps method
; https://learn.microsoft.com/en-us/windows/win32/api/rometadataapi/nf-rometadataapi-imetadataimport-gettypedefprops

HRESULT GetTypeDefProps(
  [in]                                               mdTypeDef tkTypeDef,
  [out, size_is(cchTypeDef), length_is(*pchTypeDef)] LPWSTR    szTypeDef,
  [in]                                               ULONG     cchTypeDef,
  [out]                                              ULONG     *pchTypeDef,
  [out]                                              DWORD     *pdwTypeDefFlags,
  [out]                                              mdToken   *ptkExtends
);

이 중에서 5번째 인자에 타입의 속성에 대한 정보를 받아올 수 있는데요.

CorTypeAttr Enumeration
; https://learn.microsoft.com/en-us/dotnet/framework/unmanaged-api/metadata/cortypeattr-enumeration

위의 상수 표현을 보면 tdClass 속성이 있지만 값이 0x0이라서 (참조 형식의) 클래스만 그 값을 가지고 있는지 확인할 길이 없습니다. 즉, tdStruct 또는 tdValueType과 같은 이름의 상수값이 없는 것입니다.

재미있는 것은, .NET BCL의 Type 클래스에는 IsClass라는 속성으로 해당 타입의 값/참조 형식을 구분할 수 있는 기능이 제공된다는 점입니다. 그런데, 이 값을 구해오는 방법이 흥미롭습니다. .NET Reflector로 보면 다음과 같은데,

public bool IsClass
{
    get
    {
        return (((this.GetAttributeFlagsImpl() & TypeAttributes.ClassSemanticsMask) == TypeAttributes.AnsiClass) && !this.IsValueType);
    }
}

중요한 것은 IsValueType입니다. 그리고, 놀랍게도~~~ IsValueType의 구현은 '일반적인 상식(?)'을 벗어나는 방식으로 만들어졌습니다.

public bool IsValueType
{
    get
    {
        return this.IsValueTypeImpl();
    }
}
        
internal static readonly RuntimeType ValueType = ((RuntimeType) typeof(System.ValueType));

protected virtual bool IsValueTypeImpl()
{
    return this.IsSubclassOf(RuntimeType.ValueType);
}

public virtual bool IsSubclassOf(Type c)
{
    Type baseType = this;
    if (!(baseType == c))
    {
        while (baseType != null)
        {
            if (baseType == c)
            {
                return true;
            }
            baseType = baseType.BaseType;
        }
        return false;
    }
    return false;
}

그렇습니다. ^^ 특정 타입의 값/참조 형식을 알고 싶다면 이렇게 해당 클래스의 상위로 계속 찾아들어가 System.ValueType이 나올 때까지 탐색해 보는 것입니다.

결론을 내면, 메타데이터에는 Value/Reference 여부를 알 수 있는 정보는 저장되어 있지 않습니다.




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 7/3/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)
13592정성태4/2/20241540C/C++: 165. CLion으로 만든 Rust Win32 DLL을 C#과 연동 [1]
13591정성태4/2/20241486닷넷: 2234. C# - WPF 응용 프로그램에 Blazor App 통합파일 다운로드1
13590정성태3/31/20241357Linux: 70. Python - uwsgi 응용 프로그램이 k8s 환경에서 OOM 발생하는 문제
13589정성태3/29/20241424닷넷: 2233. C# - 프로세스 CPU 사용량을 나타내는 성능 카운터와 Win32 API파일 다운로드1
13588정성태3/28/20241806닷넷: 2232. C# - Unity + 닷넷 App(WinForms/WPF) 간의 Named Pipe 통신 [2]파일 다운로드1
13587정성태3/27/20241475오류 유형: 900. Windows Update 오류 - 8024402C, 80070643
13586정성태3/27/20241811Windows: 263. Windows - 복구 파티션(Recovery Partition) 용량을 늘리는 방법
13585정성태3/26/20241581Windows: 262. PerformanceCounter의 InstanceName에 pid를 추가한 "Process V2"
13584정성태3/26/20241501개발 환경 구성: 708. Unity3D - C# Windows Forms / WPF Application에 통합하는 방법파일 다운로드1
13583정성태3/25/20241532Windows: 261. CPU Utilization이 100% 넘는 경우를 성능 카운터로 확인하는 방법
13582정성태3/19/20241674Windows: 260. CPU 사용률을 나타내는 2가지 수치 - 사용량(Usage)과 활용률(Utilization)파일 다운로드1
13581정성태3/18/20241800개발 환경 구성: 707. 빌드한 Unity3D 프로그램을 C++ Windows Application에 통합하는 방법
13580정성태3/15/20241327닷넷: 2231. C# - ReceiveTimeout, SendTimeout이 적용되지 않는 Socket await 비동기 호출파일 다운로드1
13579정성태3/13/20241530오류 유형: 899. HTTP Error 500.32 - ANCM Failed to Load dll
13578정성태3/11/20241711닷넷: 2230. C# - 덮어쓰기 가능한 환형 큐 (Circular queue)파일 다운로드1
13577정성태3/9/20242009닷넷: 2229. C# - 닷넷을 위한 난독화 도구 소개 (예: ConfuserEx)
13576정성태3/8/20241640닷넷: 2228. .NET Profiler - IMetaDataEmit2::DefineMethodSpec 사용법
13575정성태3/7/20241768닷넷: 2227. 최신 C# 문법을 .NET Framework 프로젝트에 쓸 수 있을까요?
13574정성태3/6/20241669닷넷: 2226. C# - "Docker Desktop for Windows" Container 환경에서의 IPv6 DualMode 소켓
13573정성태3/5/20241599닷넷: 2225. Windbg - dumasync로 분석하는 async/await 호출
13572정성태3/4/20241792닷넷: 2224. C# - WPF의 Dispatcher Queue로 알아보는 await 호출의 hang 현상파일 다운로드1
13571정성태3/1/20241848닷넷: 2223. C# - await 호출과 WPF의 Dispatcher Queue 동작 확인파일 다운로드1
13570정성태2/29/20241846닷넷: 2222. C# - WPF의 Dispatcher Queue 동작 확인파일 다운로드1
13569정성태2/28/20241762닷넷: 2221. C# - LoadContext, LoadFromContext 그리고 GAC파일 다운로드1
13568정성태2/27/20241887닷넷: 2220. C# - .NET Framework 프로세스의 LoaderOptimization 설정을 확인하는 방법파일 다운로드1
13567정성태2/27/20241836오류 유형: 898. .NET Framework 3.5 이하에서 mscoree.tlb 참조 시 System.BadImageFormatException파일 다운로드1
1  [2]  3  4  5  6  7  8  9  10  11  12  13  14  15  ...