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# Q&A] 제너릭 T를 인스턴스하는 방법이 뭘까요?
; https://forum.dotnetdev.kr/t/c-q-a-t/8327

제네릭의 경우, 형식 매개변수의 타입에 해당하는 인스턴스를 생성하기 위해서는 new() 제약을 사용할 수 있습니다.

namespace ConsoleApp1;

internal class Program
{
    static void Main(string[] args)
    {
        var pg = Class1<Program>.GetInstance();
        Console.WriteLine(pg); // 출력 결과: ConsoleApp1.Program
    }
}

public class Class1<T> where T: new()
{
    public static T GetInstance()
    {
        return new T();
    }
}

위의 경우, 매개변수를 갖지 않는 "기본 생성자"를 호출할 수 있도록 하는데요, 그렇다면 만약 매개변수가 있는 생성자의 호출은 어떻게 할 수 있을까요?

일단, 공식적으로는 제네릭에서 매개변수를 가진 생성자는 호출할 수 없습니다. 단지, 그것을 Reflection을 통해 우회할 수 있는데요, 예를 들어, int와 string을 받는 생성자는 다음과 같이 호출할 수 있습니다.

using System.Reflection;

namespace ConsoleApp1;

internal class Program
{
    static void Main(string[] args)
    {
        {
            var pg = Class1<Program>.GetInstance(10, "홍길동");
            Console.WriteLine(pg == null ? "(null)" : pg); // 출력 결과: (null)
        }

        {
            var pg = Class1<Person>.GetInstance(10, "홍길동");
            Console.WriteLine(pg == null ? "(null)" : pg); //  출력 결과: Person { age = 10, name = 홍길동 }
        }
    }
}

public record class Person(int age, string name)
{
    public Person() : this(0, "") { }
}

public class Class1<T> where T : class
{
    public static T? GetInstance(int age, string name)
    {
        Type t = typeof(T);

        ConstructorInfo? ctor = t.GetConstructor(new Type[] { typeof(int), typeof(string) });
        if (ctor == null)
        {
            return null;
        }

        return ctor.Invoke(new object[] { age, name }) as T;
    }
}

이 정도의 코드면, 그래도 문법적인 지원이 없는 것에 크게 아쉽지는 않을 것입니다. ^^




그나저나, AOT(Ahead-of-Time) 컴파일 기능이 많이 좋아졌나 봅니다. 예전에는 위와 같은 수준의 Reflection을 사용한 코드는 실패했던 것 같은데, 이제는 잘 동작합니다. 실제로 위의 코드를 .NET MAUI 앱으로 제작해 Android App에 AOT를 켠 Release 모드로 배포/실행이 되었습니다.




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







[최초 등록일: ]
[최종 수정일: 9/19/2023]

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

비밀번호

댓글 작성자
 




... 196  [197] 
NoWriterDateCnt.TitleFile(s)
17정성태7/17/200327458VC++: 6. Win32 API Hook - 소스는 "공개소스"에있습니다. [2]
16정성태7/17/200319791COM 개체 관련: 3. IE 툴밴드의 위치문제파일 다운로드1
15정성태7/17/200320797VC++: 5. 시행착오 - 클래스 포인터를 void * 로 대입후 delete 하는 경우.
14정성태7/17/200322326VC++: 4. MFC Message 처리 구조
13정성태7/17/200322083VC++: 3. template 활용의 최고 단계!
12정성태5/7/200619639VC++: 2. void func1( MYCLASS *&pBuildingElement ); 선언의 의미
10정성태7/17/200317982기타: 2. 데브피아 사이트의 클럽 서비스 내에 있는 Standard C++ Research
9정성태7/17/200323549기타: 1. Programming Applications for Microsoft Windows 4th Edition
8정성태7/17/200320942COM 개체 관련: 2. CWindowImpl 의 기본 윈도우 클래스 명을 바꾸려면?
7정성태7/17/200323520VS.NET IDE: 1. VC++ 프로파일링 사용법
4정성태7/17/200335900VC++: 1. C++ 클래스 멤버변수 초기화 [3]
3정성태7/17/200321865스크립트: 2. JScript에서의 Blocking 동작을 막아주는 COM 메서드
2정성태6/14/200640642COM 개체 관련: 1. IWebBrowser2와 IHTMLDocument2의 상호 변환 [2]
1정성태7/17/200324807스크립트: 1. IE 의 훌륭한 저장기능 - userData
... 196  [197]