Eri*_*ter 3 c# generics extension-methods types dynamic
我正在尝试执行一个返回 T 类型对象的扩展方法,但我正在尝试基于 Header/Detail 动态泛型类型使类型 T 动态化。
这可能有点冗长...
using System;
using System.Collections.Generic;
namespace Blah
{
public interface IHeader
{
string Name { get; set; }
IDetail Detail { get; set; }
}
public interface IDetail
{
//Nothing the 'Real' implementation of this
//interface will have it's own properties.
}
public class GenericHeader : IHeader
{
public string Name { get; set; }
public IDetail Detail { get; set; }
}
public class RealHeader : GenericHeader
{
public new RealDetail Detail
{
get { return (RealDetail) base.Detail; }
set { base.Detail = value; }
}
}
public class RealDetail : IDetail
{
public string ThisImportantOnlyToRealDetail { get; set; }
}
public static class ExtensionHelpers
{
public static T ToObject<T>(this IDictionary<string, string> reader) where T : new()
{
//This maps the dictionary to Key Value Pairs of the Object's properties
//it then will return a object filled with the values
return new T();
}
}
public class MyRepo<THeader> where THeader : class, IHeader, new()
{
public THeader GetById(int ID)
{
THeader returnHeader = new THeader();
//Process to fill Header
var dictDetail = new Dictionary<string, string>();
//Process to fill Detail Dictionary
//Use extension method to create an Object
//based on Key Value Pairs from Dictionary
// !!!!!!!!!!!!!!!! This Is The Problem !!!!!!!!!!!!!!!!
// Can't use typeof for returnHeader.Detail, reflection?
returnHeader.Detail = dictDetail.ToObject<typeof(returnHeader.Detail)>();
return returnHeader;
}
}
public class Worker
{
public void DoWork()
{
var myRealRepo = new MyRepo<RealHeader>();
var myRealHeader = myRealRepo.GetById(123);
Console.WriteLine(myRealHeader.Detail.ThisImportantOnlyToRealDetail);
}
}
}
Run Code Online (Sandbox Code Playgroud)
这必须使用反射来完成。
typeof(ExtensionHelpers)
.GetMethod("ToObject", BindingFlags.Static | BindingFlags.Public)
.MakeGenericMethod(returnHeader.Detail.GetType())
.Invoke(null, new object[] { dictDetail });
Run Code Online (Sandbox Code Playgroud)
请注意,由于扩展方法是一种语言功能,因此在使用反射时必须像调用常规静态方法一样调用该方法。
如果类型始终是动态的,那么将您更改ToObject为将 aType作为参数的常规非泛型方法可能会更容易。
您的设计实际上有点问题,因为您似乎需要知道Detail属性背后的对象的实际类型,但这要求该属性已经有一个值,但您的代码正在设置该属性。
我建议你考虑其他方法来处理这个问题。