Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (seongtaejeong at gmail.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)
(시리즈 글이 3개 있습니다.)
.NET Framework: 408. 자바와 닷넷의 제네릭 차이점 - 중간 언어 및 공변/반공변 처리
; https://www.sysnet.pe.kr/2/0/1581

.NET Framework: 743. C# 언어의 공변성과 반공변성
; https://www.sysnet.pe.kr/2/0/11513

닷넷: 2264. C# - 형식 인자로 인터페이스를 갖는 제네릭 타입으로의 형변환
; https://www.sysnet.pe.kr/2/0/13636




C# - 형식 인자로 인터페이스를 갖는 제네릭 타입으로의 형변환

짧은 제목으로 표현하기 좀 힘들군요. ^^; 예를 들어, 아래와 같이 인터페이스를 상속받는 타입의 경우,

public interface IData
{
}

public class DataImpl : IData { }

때로는 저 DataImpl 타입을 반환하는 제네릭 타입이 있을 것입니다.

static private Task<DataImpl> GetDataImplAsync()
{
    // Task.Run으로 해도 되지만, 테스트 프로젝트를 .NET Framework 4.0으로 맞추느라 Task.Factory를 사용
    Task<DataImpl> task = Task.Factory.StartNew<DataImpl>(() =>
    {
        DataImpl data = new DataImpl();
        return data;
    });

    return task;
}

문제는, 저런 경우 IData 인터페이스로 공통 처리할 수 있는 방법이 없습니다. 즉, 다음과 같이 인터페이스가 아닌, 구현 타입을 갖는 제네릭을 받아야 합니다. (이에 대해서는 전에 자세히 다룬 글이 있고, C# 4부터 인터페이스에 한해 공변을 지원하는 보완을 했습니다.)

// error CS0029: Cannot implicitly convert type 'System.Threading.Tasks.Task<DataImpl>' to 'System.Threading.Tasks.Task<IData>'
// Task<IData> data2 = GetDataImplAsync();

Task<DataImpl> data = GetDataImplAsync();

저런 제약 때문에 DataImpl 구현 클래스를 담은 어셈블리를 반드시 참조하거나, 아니면 Reflection을 써서 접근을 해야 합니다. 물론 이런 경우 Reflection보다는 dynamic을 쓰는 것이 더 좋은 선택입니다.

dynamic data = GetDataImplAsync();
IData iData = data.Result; // 인터페이스로 형변환




만약 dynamic을 쓰고 싶지 않다면 어떻게 해야 할까요? 어쩔 수 없이 그럴 때는 Reflection으로 접근해야 합니다. 그런데, 다시 한번 만약, ^^ Reflection 속도보다 빠르게 접근하고 싶다면 어떻게 해야 할까요? 이럴 때는 (역시 예전에 dynamic과 비교해 설명한) LCG를 사용하면 됩니다.

그래서 대충 이런 식으로 만들어 주면,

private static Dictionary<string, Func<object, object>> _taskResultFunc = new();

private static object GetDynamicResult(object objValue)
{
    Type type = objValue.GetType();

    if (_taskResultFunc.TryGetValue(type.FullName, out var resultFunc) == false)
    {
        PropertyInfo prop = type.GetProperty("Result");
        MethodInfo mi = prop.GetGetMethod();

        DynamicMethod dm = new DynamicMethod(
            Guid.NewGuid().ToString(),
            typeof(object), [typeof(object)], typeof(Program), false);

        ILGenerator il = dm.GetILGenerator();
        il.Emit(OpCodes.Ldarg_0);
        il.Emit(OpCodes.Isinst, type);
        il.Emit(OpCodes.Callvirt, mi);
        il.Emit(OpCodes.Ret);

        resultFunc = (Func<object, object>)dm.CreateDelegate(typeof(Func<object, object>));
        _taskResultFunc[type.FullName] = resultFunc;
    }

    return resultFunc(objValue);
}

이후 대상을 직접 참조하지 않고도 인터페이스 형변환만으로 다룰 수 있습니다.

IData result = GetDynamicResult(objData) as IData;

설령 이렇게 구현 클래스가 추가돼도,

public class AttrImpl : IData { }

static private Task<AttrImpl> GetAttrImplAsync()
{
    Task<AttrImpl> task = Task.Factory.StartNew<AttrImpl>(() =>
    {
        AttrImpl data = new AttrImpl();
        return data;
    });

    return task;
}

마찬가지의 방식으로 다룰 수 있습니다.

object objAttr = GetAttrImplAsync();
result = GetDynamicResult(objAttr) as IData;

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 6/17/2024]

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

비밀번호

댓글 작성자
 




... 106  107  108  109  110  [111]  112  113  114  115  116  117  118  119  120  ...
NoWriterDateCnt.TitleFile(s)
11182정성태4/19/201724636개발 환경 구성: 313. Nuget Facebook 라이브러리를 이용해 ASP.NET 웹 폼과 로그인 연동하는 방법
11181정성태4/18/201721800개발 환경 구성: 312. Azure Web Role의 AppPool 실행 권한을 Local System으로 바꾸는 방법
11180정성태4/16/201724933Java: 18. Java의 Memory Mapped File 자원 반환이 안 되는 문제
11179정성태4/13/201717914기타: 64. SVG Converter 스토어 앱 개인정보 보호 정책 안내
11178정성태4/10/201720201개발 환경 구성: 311. COM+ 관리자의 DCOM 구성에 나오는 기준
11177정성태4/7/201720353.NET Framework: 653. C# 7 새로운 문법(1) - 더욱 편리해진 Out 변수 사용파일 다운로드1
11176정성태4/5/201717490VC++: 117. Visual Studio - ATL COM 개체를 단위 테스트 하는 방법
11175정성태4/5/201726749.NET Framework: 652. C# 개발자를 위한 C++ COM 객체의 기본 구현 방식 설명파일 다운로드1
11174정성태4/3/201720578VC++: 116. Visual Studio 단위 테스트 - Failed to set up the execution context to run the test
11173정성태4/3/201723734VC++: 115. Visual Studio에서 C++ DLL을 대상으로 단위 테스트할 때 비정상 종료한다면?파일 다운로드1
11172정성태4/3/201722770.NET Framework: 651. C# - 특정 EXE 프로세스를 종료시킨 EXE를 찾아내는 방법파일 다운로드1
11171정성태3/31/201719929VS.NET IDE: 114. Visual Studio 디버깅 경고 창 - You are debugging a Release build of ...
11170정성태3/31/201722279.NET Framework: 650. C# - CachedAnonymousMethodDelegate 유형의 코드 생성
11169정성태3/30/201721855VC++: 114. C++ vtable의 가상 함수 호출 가로채기파일 다운로드1
11168정성태3/29/201724565VC++: 113. C++ 클래스 상속 관계의 vtable 생성 과정
11167정성태3/28/201724998VC++: 112. C++의 가상 함수 테이블 (vtable)은 언제 생성될까요? [2]
11166정성태3/28/201719958오류 유형: 382. System.Data.SqlClient.SqlException - Arithmetic overflow error converting IDENTITY to data type int.
11165정성태3/27/201723101오류 유형: 381. Visual C++에서 min, max 함수를 사용한 경우 C2589, C2059 컴파일 오류 발생
11164정성태3/27/201731572VC++: 111. C++ 클래스의 상속에 따른 메모리 구조 [2]파일 다운로드1
11163정성태3/25/201721226VC++: 110. CreateThread Win32 API에 C++ 클래스의 멤버 함수를 전달하는 방법파일 다운로드1
11162정성태3/24/201725505오류 유형: 380. Visual Studio 빌드 실패 - The OutputPath property is not set for project
11161정성태3/24/201717355오류 유형: 379. ICOMAdminCatalog.GetCollection 호출 시 0x80070422 예외 발생
11160정성태3/23/201723165.NET Framework: 649. ASP.NET - Server cannot append header after HTTP headers have been sent. (HTTP 헤더를 보낸 후에는 서버에서 헤더를 추가할 수 없습니다.)파일 다운로드1
11159정성태3/23/201720465Windows: 136. Memory-mapped File은 Private Bytes 크기에 포함될까요?파일 다운로드1
11158정성태3/22/201719356디버깅 기술: 85. Windbg - SOS 디버깅 사례 System.NullReferenceException 예외 추적
11157정성태3/22/201722691.NET Framework: 648. Dictionary<TKey, TValue>를 deep copy하는 방법파일 다운로드1
... 106  107  108  109  110  [111]  112  113  114  115  116  117  118  119  120  ...