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;
(
첨부 파일은 이 글의 예제 코드를 포함합니다.)
[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]