我需要在其中使用以下通用类和方法ParseFrom()
:
public sealed class MessageParser<T> : MessageParser where T : IMessage<T>
{
public MessageParser(Func<T> factory); //constructor
public T ParseFrom(byte[] data);
}
现在,我不知道这个类在编译时的参数类型,所以我使用类型反射和MakeGenericType()
方法来做到这一点:
//Assuming itemInstance is given as input parameter
Type typeArgument = itemInstance.GetType();
Type genericClass = typeof(MessageParser<>);
var genericType = genericClass.MakeGenericType(typeArgument);
var instance = Activator.CreateInstance(genericType);
它给了我一个运行时错误:没有无参数构造函数。但是当我尝试将Func<T> factory
作为CreateInstance()
的参数传递时:
var instance = Activator.CreateInstance(genericType, () => Activator.CreateInstance(typeArgument));
它给了我一个编译错误:无法将lambda表达式转换为类型'字符串',因为它不是委托类型。我是否在这里使用了错误的委托函数语法?
动态构造未知类型的委托不像使用反射调用方法那么简单,所以最简单的选择是编写一个静态类型的方法来构造委托,然后使用反射调用它。
public class DelegateCreator
{
public static Func<T> MakeConstructorStatically<T>()
{
return Activator.CreateInstance<T>;
}
public static object MakeConstructorDynamically(Type type)
{
return typeof(DelegateCreator)
.GetMethod(nameof(MakeConstructorStatically))
.MakeGenericMethod(type)
.Invoke(null, Array.Empty<object>());
}
}
通过反射创建Func<T>
,CreateDelegate
是可行的方法。因此,需要一个具有预期签名的方法——包括类型约束(T是IMessage
你可以这样做。缺点是,您仍然需要使用反射来调用解析器的方法,至少是那些使用类型参数的方法:
public class CreateParserLateBound {
//The method with the matching signature
public static T MessageParserFactory<T>()
where T : IMessage<T>
{
//your factory code, you pass to MessageParser(Func<T> factory) goes here...
return default(T);
}
...
// itemInstance == item that is IMesage<T>, with T unknown at compiletime;
var itemType = itemInstance.GetType();
var boundParserType = typeof(MessageParser<>).MakeGenericType(itemType);
var boundFuncType = typeof(Func<>).MakeGenericType(itemType);
var factoryMethodInstance = typeof(CreateParserLateBound )
.GetMethod("MessageParserFactory")
.MakeGenericMethod(itemType)
.CreateDelegate(boundFuncType);
var parserInstance = Activator.CreateInstance(boundParserType,
new object[]{ factoryMethodInstance } );
//Invoke ParseFrom (also through reflection)
byte[] data = {1,2,3,4};
boundParserType.InvokeMember("ParseFrom",
BindingFlags.Public | BindingFlags.Instance | BindingFlags.InvokeMethod, null,
parserInstance, new object[] {data});
完整可运行代码@ https://dotnetfiddle.net/RIOEXA
简单的答案是编写自己的泛型方法,然后通过反射调用它。
public static class Foo
{
public static MessageParser<T> CreateParser<T>() where T : IMessage<T>, new()
=> new MessageParser<T>(() => new T());
private static MethodInfo _createMethod = typeof(Foo)
.GetMethods()
.Where(m => m.Name == nameof(CreateParser) && m.IsGenericMethod)
.Single();
public static MessageParser CreateParser(Type type)
=> (MessageParser)_createMethod.MakeGenericMethod(type)
.Invoke(null, new object[] { });
}