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

비밀번호

댓글 작성자
 




... 31  32  33  [34]  35  36  37  38  39  40  41  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
13123정성태9/8/202217172.NET Framework: 2046. C# 11 - 멤버(속성/필드)에 지정할 수 있는 required 예약어 추가
13122정성태8/26/202217571.NET Framework: 2045. C# 11 - 메서드 매개 변수에 대한 nameof 지원
13121정성태8/23/202213071C/C++: 157. Golang - 구조체의 slice 필드를 Reflection을 이용해 변경하는 방법
13120정성태8/19/202217635Windows: 209. Windows NT Service에서 UI를 다루는 방법 [3]
13119정성태8/18/202216718.NET Framework: 2044. .NET Core/5+ 프로젝트에서 참조 DLL이 보관된 공통 디렉터리를 지정하는 방법
13118정성태8/18/202213492.NET Framework: 2043. WPF Color의 기본 색 영역은 (sRGB가 아닌) scRGB [2]
13117정성태8/17/202217990.NET Framework: 2042. C# 11 - 파일 범위 내에서 유효한 타입 정의 (File-local types)파일 다운로드1
13116정성태8/4/202218368.NET Framework: 2041. C# - Socket.Close 시 Socket.Receive 메서드에서 예외가 발생하는 문제파일 다운로드1
13115정성태8/3/202219028.NET Framework: 2040. C# - ValueTask와 Task의 성능 비교 [1]파일 다운로드1
13114정성태8/2/202218743.NET Framework: 2039. C# - Task와 비교해 본 ValueTask 사용법파일 다운로드1
13113정성태7/31/202218380.NET Framework: 2038. C# 11 - Span 타입에 대한 패턴 매칭 (Pattern matching on ReadOnlySpan<char>)
13112정성태7/30/202219141.NET Framework: 2037. C# 11 - 목록 패턴(List patterns) [1]파일 다운로드1
13111정성태7/29/202218617.NET Framework: 2036. C# 11 - IntPtr/UIntPtr과 nint/nuint의 통합파일 다운로드1
13110정성태7/27/202217862.NET Framework: 2035. C# 11 - 새로운 연산자 ">>>" (Unsigned Right Shift)파일 다운로드1
13109정성태7/27/202220010VS.NET IDE: 177. 비주얼 스튜디오 2022를 이용한 (소스 코드가 없는) 닷넷 모듈 디버깅 - "외부 원본(External Sources)" [1]
13108정성태7/26/202217168Linux: 53. container에 실행 중인 Golang 프로세스를 디버깅하는 방법 [1]
13107정성태7/25/202215978Linux: 52. Debian/Ubuntu 계열의 docker container에서 자주 설치하게 되는 명령어
13106정성태7/24/202214897오류 유형: 819. 닷넷 6 프로젝트의 "Conditional compilation symbols" 기본값 오류
13105정성태7/23/202218258.NET Framework: 2034. .NET Core/5+ 환경에서 (프로젝트가 아닌) C# 코드 파일을 입력으로 컴파일하는 방법 - 두 번째 이야기 [1]
13104정성태7/23/202221851Linux: 51. WSL - init에서 systemd로 전환하는 방법
13103정성태7/22/202217185오류 유형: 818. WSL - systemd-genie와 관련한 2가지(systemd-remount-fs.service, multipathd.socket) 에러
13102정성태7/19/202216771.NET Framework: 2033. .NET Core/5+에서는 구할 수 없는 HttpRuntime.AppDomainAppId
13101정성태7/15/202230315도서: 시작하세요! C# 10 프로그래밍
13100정성태7/15/202218858.NET Framework: 2032. C# 11 - shift 연산자 재정의에 대한 제약 완화 (Relaxing Shift Operator)
13099정성태7/14/202218181.NET Framework: 2031. C# 11 - 사용자 정의 checked 연산자파일 다운로드1
13098정성태7/13/202215879개발 환경 구성: 647. Azure - scale-out 상태의 App Service에서 특정 인스턴스에 요청을 보내는 방법 [1]
... 31  32  33  [34]  35  36  37  38  39  40  41  42  43  44  45  ...