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 모드로 배포/실행이 되었습니다.
[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]