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

비밀번호

댓글 작성자
 




... 76  77  78  [79]  80  81  82  83  84  85  86  87  88  89  90  ...
NoWriterDateCnt.TitleFile(s)
11961정성태6/27/201917829Graphics: 37. C# - PLplot - 출력 모음(Family File Output)
11960정성태6/27/201918892Graphics: 36. C# - PLplot의 16색 이상을 표현하는 방법과 subpage를 이용한 그리드 맵 표현
11959정성태6/27/201920061Graphics: 35. matplotlib와 PLplot의 한글 처리
11958정성태6/25/201924569Linux: 18. C# - .NET Core Console로 리눅스 daemon 프로그램 만드는 방법 [6]
11957정성태6/24/201922919Windows: 160. WMI 쿼리를 명령행에서 간단하게 수행하는 wmic.exe [2]
11956정성태6/24/201921416Linux: 17. CentOS 7에서 .NET Core Web App 실행 환경 구성 [1]
11955정성태6/20/201919750Math: 60. C# - 로지스틱 회귀를 이용한 분류파일 다운로드1
11954정성태6/20/201918511오류 유형: 550. scp - sudo: no tty present and no askpass program specified
11953정성태6/20/201916689오류 유형: 549. The library 'libhostpolicy.so' required to execute the application was not found in '...'
11952정성태6/20/201917380Linux: 16. 우분투, Centos의 Netbios 호스트 이름 풀이 방법
11951정성태6/20/201920595오류 유형: 548. scp 연결 시 "Permission denied" 오류 및 "WARNING: REMOTE HOST IDENTIFICATION HAS CHANGED!" 경고
11950정성태6/18/201920746.NET Framework: 845. C# - 윈도우 작업 관리자와 리소스 모니터의 메모리 값을 구하는 방법
11949정성태6/18/201916057오류 유형: 547. CoreCLR Profiler 예제 프로젝트 빌드 시 컴파일 오류 유형
11948정성태6/17/201918496Linux: 15. 리눅스 환경의 Visual Studio Code에서 TFS 서버 연동
11947정성태6/17/201920266Linux: 14. 리눅스 환경에서 TFS 서버 연동
11946정성태6/17/201921247개발 환경 구성: 445. C# - MathNet으로 정규 분포를 따르는 데이터를 생성, PLplot으로 Histogram 표현파일 다운로드1
11945정성태6/17/201918976Linux: 13. node.js에서 syslog로 출력하는 방법
11944정성태6/16/201925345Linux: 12. Ubuntu 16.04/18.04에서 node.js 최신 버전 설치 방법
11943정성태6/15/201918576.NET Framework: 844. C# - 박싱과 언박싱 [1]
11942정성태6/13/201924825개발 환경 구성: 444. 로컬의 Visual Studio Code로 원격 리눅스 머신에 접속해 개발하는 방법 [1]
11941정성태6/13/201917507오류 유형: 546. "message NETSDK1057: You are using a preview version of .NET Core" 빌드 경고 없애는 방법
11940정성태6/13/201917753개발 환경 구성: 443. Visual Studio의 Connection Manager 기능(Remote SSH 관리)을 위한 명령행 도구파일 다운로드1
11939정성태6/13/201916532오류 유형: 545. Managed Debugging Assistant 'FatalExecutionEngineError'
11938정성태6/12/201919064Math: 59. C# - 웨이트 벡터 갱신식을 이용한 퍼셉트론 분류파일 다운로드1
11937정성태6/11/201925395개발 환경 구성: 442. .NET Core 3.0 preview 5를 이용해 Windows Forms/WPF 응용 프로그램 개발 [1]
11936정성태6/10/201918346Math: 58. C# - 최소 자승법의 1차, 2차 수렴 그래프 변화 확인 [2]파일 다운로드1
... 76  77  78  [79]  80  81  82  83  84  85  86  87  88  89  90  ...