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

비밀번호

댓글 작성자
 




... 16  17  18  19  20  21  22  23  24  25  26  27  28  29  [30]  ...
NoWriterDateCnt.TitleFile(s)
13189정성태12/9/202216907오류 유형: 831. Visual Studio - Windows Forms 디자이너의 도구 상자에 컨트롤이 보이지 않는 문제
13188정성태12/9/202215374.NET Framework: 2075. C# - 직접 만들어 보는 TaskScheduler 실습 (SingleThreadTaskScheduler)파일 다운로드1
13187정성태12/8/202215346개발 환경 구성: 654. openssl - CA로부터 인증받은 새로운 인증서를 생성하는 방법 (2)
13186정성태12/6/202213121오류 유형: 831. The framework 'Microsoft.AspNetCore.App', version '...' was not found.
13185정성태12/6/202214076개발 환경 구성: 653. Windows 환경에서의 Hello World x64 어셈블리 예제 (NASM 버전)
13184정성태12/5/202212646개발 환경 구성: 652. ml64.exe와 link.exe x64 실행 환경 구성 [1]
13183정성태12/4/202212474오류 유형: 830. MASM + CRT 함수를 사용하는 경우 발생하는 컴파일 오류 정리 [1]
13182정성태12/4/202214281Windows: 217. Windows 환경에서의 Hello World x64 어셈블리 예제 (MASM 버전)
13181정성태12/3/202212797Linux: 54. 리눅스/WSL - hello world 어셈블리 코드 x86/x64 (nasm)
13180정성태12/2/202213608.NET Framework: 2074. C# - 스택 메모리에 대한 여유 공간 확인하는 방법파일 다운로드1
13179정성태12/2/202212326Windows: 216. Windows 11 - 22H2 업데이트 이후 Terminal 대신 cmd 창이 뜨는 경우
13178정성태12/1/202213414Windows: 215. Win32 API 금지된 함수 - IsBadXxxPtr 유의 함수들이 안전하지 않은 이유파일 다운로드1
13177정성태11/30/202213996오류 유형: 829. uwsgi 설치 시 fatal error: Python.h: No such file or directory
13176정성태11/29/202211538오류 유형: 828. gunicorn - ModuleNotFoundError: No module named 'flask'
13175정성태11/29/202215835오류 유형: 827. Python - ImportError: cannot import name 'html5lib' from 'pip._vendor'
13174정성태11/28/202212541.NET Framework: 2073. C# - VMMap처럼 스택 메모리의 reserve/guard/commit 상태 출력파일 다운로드1
13173정성태11/27/202213468.NET Framework: 2072. 닷넷 응용 프로그램의 스레드 스택 크기 변경
13172정성태11/25/202212971.NET Framework: 2071. 닷넷에서 ESP/RSP 레지스터 값을 구하는 방법파일 다운로드1
13171정성태11/25/202212499Windows: 214. 윈도우 - 스레드 스택의 "red zone"
13170정성태11/24/202213413Windows: 213. 윈도우 - 싱글 스레드는 컨텍스트 스위칭이 없을까요?
13169정성태11/23/202214707Windows: 212. 윈도우의 Protected Process (Light) 보안 [1]파일 다운로드2
13168정성태11/22/202212601제니퍼 .NET: 31. 제니퍼 닷넷 적용 사례 (9) - DB 서비스에 부하가 걸렸다?!
13167정성태11/21/202213059.NET Framework: 2070. .NET 7 - Console.ReadKey와 리눅스의 터미널 타입
13166정성태11/20/202213006개발 환경 구성: 651. Windows 사용자 경험으로 WSL 환경에 dotnet 런타임/SDK 설치 방법
13165정성태11/18/202212011개발 환경 구성: 650. Azure - "scm" 프로세스와 엮인 서비스 모음
13164정성태11/18/202213937개발 환경 구성: 649. Azure - 비주얼 스튜디오를 이용한 AppService 원격 디버그 방법
... 16  17  18  19  20  21  22  23  24  25  26  27  28  29  [30]  ...