Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

(시리즈 글이 4개 있습니다.)
.NET Framework: 784. C# - 제네릭 인자를 가진 타입을 생성하는 방법
; https://www.sysnet.pe.kr/2/0/11582

.NET Framework: 787. object로 형변환된 인스턴스를 원래의 타입 인자로 제네릭 메서드를 호출하는 방법
; https://www.sysnet.pe.kr/2/0/11589

닷넷: 2145. C# - 제네릭의 형식 매개변수에 속한 (매개변수를 가진) 생성자를 호출하는 방법
; https://www.sysnet.pe.kr/2/0/13417

닷넷: 2251. C# - 제네릭 인자를 가진 타입을 생성하는 방법 - 두 번째 이야기
; https://www.sysnet.pe.kr/2/0/13610




C# - 제네릭 인자를 가진 타입을 생성하는 방법 - 두 번째 이야기

예를 들어, 다음과 같은 C# 코드가 있을 때,

ValueTuple<int, int, int> value = new(5, 6, 7);

저 value에 대해서 Reflection으로 5, 6, 7 값을 가져오고 싶다면 어떻게 해야 할까요?

우선, ValueTuple<T1, T2, T3>에 대한 타입 정보를 가져와야 하는데요, C# 코드에서 사용하듯이 이렇게 하시면,

Type stateType = typeof(AppDomain).Assembly.GetType("System.ValueTuple<int, int, int>");

// 또는,

Type stateType = typeof(AppDomain).Assembly.GetType("System.ValueTuple<T1, T2, T3>");

타입 정보를 찾을 수 없습니다. 이런 경우 특별한 표기법을 이용해야 하는데요, backtick(`)을 이용해 제네릭 인자 수를 기입하는 식이어야 합니다.

Type stateType = typeof(AppDomain).Assembly.GetType("System.ValueTuple`3");

위의 경우 제네릭 인자의 타입을 지정하지 않았는데요, 물론 지정하는 것도 가능합니다. 그리고 이 이름은 다음과 같은 식으로 구하는 것도 가능합니다.

ValueTuple<int, int, int> value = new(5, 6, 7);
Console.WriteLine(value.GetType().FullName);
// 출력 결과:
// System.ValueTuple`3[[System.Int32, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089],[System.Int32, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089],[System.Int32, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089]]

제네릭 인자의 타입으로 int에 대한 FullName이 기록되고 있는데, 이것은 다음과 같이 줄이는 것도 가능합니다.

Type stateType = typeof(AppDomain).Assembly.GetType("System.ValueTuple`3[[System.Int32],[System.Int32],[System.Int32]]");

일단, 위와 같이 제네릭 인자의 타입까지 지정했다면 다음과 같이 해당 타입의 필드를 역시 Reflection으로 가져올 수 있습니다.

{
    var fi = stateType.GetField("Item1");
    Console.WriteLine(fi.GetValue(value)); // 출력 결과: 5
}




여기서 주의할 것은, 제네릭 인자를 지정하지 않은 경우에는 GetField로 값을 가져올 수 없다는 점입니다.

ValueTuple<int, int, int> value = new(5, 6, 7);

Type stateType = typeof(AppDomain).Assembly.GetType("System.ValueTuple`3");

var fi = stateType.GetField("Item1");
Console.WriteLine(fi.GetValue(value)); // 예외 발생
/*
Unhandled Exception: System.InvalidOperationException: Late bound operations cannot be performed on fields with types for which Type.ContainsGenericParameters is true.
   at System.Reflection.RtFieldInfo.InternalGetValue(Object obj, StackCrawlMark& stackMark)
   at System.Reflection.RtFieldInfo.GetValue(Object obj)
   at ConsoleApp2.Program.Main(String[] args)
*/

위의 오류를 고치는 방법은, 1) 이전에 설명한 것처럼 제네릭 인자를 명시해 GetType을 호출하든가, 2) 아니면 MakeGenericType을 이용해 제네릭 인자의 타입 정보를 명시해 구한 그 타입을 다시 사용해야 합니다.

ValueTuple<int, int, int> value = new(5, 6, 7);

Type stateType = typeof(AppDomain).Assembly.GetType("System.ValueTuple`3");
stateType = stateType.MakeGenericType(typeof(int), typeof(int), typeof(int));

var fi = stateType.GetField("Item1");
Console.WriteLine(fi.GetValue(value)); // 출력 결과: 5

2가지 방법의 장단점이 있는데요, 첫 번째 방법은 제네릭 인자의 타입을 정의한 어셈블리를 참조해야만 하는 반면, 두 번째 방법은 참조 없이도 그 타입 조차도 Reflection으로 접근해 MakeGenericType의 인자로 넘겨줄 수 있습니다.




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







[최초 등록일: ]
[최종 수정일: 4/28/2024]

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

비밀번호

댓글 작성자
 




... 121  122  123  124  125  126  [127]  128  129  130  131  132  133  134  135  ...
NoWriterDateCnt.TitleFile(s)
10839정성태8/22/201532882Windows: 112. 윈도우 10에서 터치 키보드를 안 뜨게 할 수 있는 방법 [5]
10838정성태8/22/201543830오류 유형: 304. Windows 10에서 VPN 연결이 실패한다면? [3]
10837정성태8/21/201521583오류 유형: 303. Your computer is low on memory. Save your files and close these programs...
10836정성태8/21/201524520오류 유형: 302. 설치 파일 실행 시 "This app can't run on your PC" 오류가 뜬다면?
10835정성태8/21/201531767웹: 31. Microsoft Edge 브라우저를 명령행에서 띄우는 방법 [1]
10834정성태8/19/201525318.NET Framework: 526. 닷넷 - 값 형식을 new 없이 생성하면 0으로 초기화되지 않는다?
10833정성태8/18/201529593.NET Framework: 525. C# - 닷넷에서 프로세스가 열고 있는 파일 목록을 구하는 방법파일 다운로드1
10832정성태8/17/201533215디버깅 기술: 74. x64 콜 스택 인자 추적과 windbg의 Child-SP, RetAddr, Args to Child 값 확인 [8]파일 다운로드2
10831정성태8/13/201534831.NET Framework: 524. .NET 4.0과 .NET 4.5의 컴파일 결과 차이점 [1]파일 다운로드1
10830정성태8/12/201528214개발 환경 구성: 275. Web.config이 적용되지 않는 프로젝트에서 Razor 템플릿 파일의 C# 컴파일러 버전 제어 [1]
10829정성태8/10/201530388개발 환경 구성: 274. PowerShell/명령행에서 JDK/JRE를 무인(unattended)/자동 설치를 하는 방법 [3]
10828정성태8/10/201535736웹: 30. Edge 브라우저에서 "이 웹 사이트에는 Internet Explorer가 필요함" 단계를 없애는 방법 [1]
10827정성태7/8/201535895개발 환경 구성: 273. Visual Studio 2015에서 Github와 연동하는 방법 [3]
10826정성태7/8/201527395오류 유형: 301. The trust relationship between this workstation and the primary domain failed. - 두 번째 이야기
10825정성태7/8/201524650개발 환경 구성: 272. Visual Studio IDE 설치 없이 Visual Studio SDK 설치하는 방법
10824정성태7/7/201530884개발 환경 구성: 271. Team Foundation Server 2015 설치 방법 [1]
10823정성태7/7/201532310오류 유형: 300. SqlException (0x80131904): Unable to open the physical file
10822정성태7/7/201530260오류 유형: 299. The 'Visual C++ Project System Package' package did not load correctly.
10821정성태7/7/201523973오류 유형: 298. Unable to start debugging on the web server. IIS does not list a web site that matches the launched URL.
10820정성태7/7/201529457오류 유형: 297. HTTP Error 503. The service is unavailable. - 두 번째
10819정성태7/2/201532402오류 유형: 296. SQL Server Express 시작 오류 - error code 3417
10818정성태7/1/201532088오류 유형: 295. HTTP Error 503. The service is unavailable. [1]
10817정성태6/29/201537064.NET Framework: 523. C# 람다(Lambda)에서 변수 캡처 방식 [3]
10816정성태6/25/201531333.NET Framework: 522. 닷넷의 어셈블리 서명 데이터 확인 방법파일 다운로드1
10815정성태6/23/201530361Graphics: 1. 자네 나와 함께... UNITY 하지 않겠는가! [4]
10814정성태6/22/201528275.NET Framework: 521. Roslyn을 이용해 C# 문법 변형하기 (2) [5]
... 121  122  123  124  125  126  [127]  128  129  130  131  132  133  134  135  ...