我有一本字典,我正在使用它来促进一些基于 api 版本号的内部路由。本质上发生的事情是我在字典中查找和操作并尝试调用它的 RUN 方法。但是为了做到这一点,我需要能够将对象转换为它的接口(interface)。这就是我的意思,这是字典:
public Dictionary<string, Type> Routing = new Dictionary<string, Type>();
public VersionRouter()
{
Routing.Add("1.0", typeof(OperationV1<RequestObjectV1, ResponseObjectV1>));
Routing.Add("2.0", typeof(OperationV1<RequestObjectV2, ResponseObjectV1>));
Routing.Add("3.0", typeof(OperationV1<RequestObjectV2, ResponseObjectV2>));
}
我可以像这样获取我想要实例化的正确类型:
var myOperation = Routing["2.0"];
然后在正常情况下,我会像这样实例化和转换它:
var myInstance = (MyInterface) Activator.CreateInstance(myOperation);
但是,接口(interface)是通用的,因为它需要知道请求和响应类型是什么:
var myInstance = (MyInterface<TRequest, TResponse>) Activator.CreateInstance(myOperation);
在这个阶段我不知道如何告诉它那些请求和响应类型是什么。我假设它可以通过反射(reflection)来完成。我发现我可以通过类似 myOperation.GetGenericArguments() 的方法获取这些通用参数,但我不确定在这个阶段如何利用它来发挥我的优势。有谁知道如何做到这一点?
最佳答案
为了扩展 SLaks 答案:
没有合乎逻辑的方式来处理您的场景。您要做的是让一段代码在运行时处理不同的类型。这只能通过构建单独的代码分支(可以完成)或回退到动态/反射来完成。澄清这一点:
class Apple { }
class Pear { }
void Handle(object what)
{
// either
if (what is Apple) {}
else if (what is Pear) {}
// or
dynamic x = what;
x.LetsHopeThisMethodExists();
// or
what.GetType().GetMethod('lorem').Invoke(what, null);
}
现在,我们可以坚持 SLak 的提议,即为 Apple
和 Pear
声明一个基类型,即 Fruit
。这样,Handle
可以接受一个 Fruit
并对 Apple
和 Pear
上可用的通用功能执行逻辑
这就引出了一个问题,如何使用泛型来做到这一点。默认情况下,泛型不支持变体,但在 .NET 4.0 中,这当然是可能的。您可以通过在类型参数上应用 out
关键字来将接口(interface)(但仅是接口(interface))声明为协变的。这使您可以执行以下操作:
interface IFruit { }
interface IBasket<out TFruit> where TFruit : IFruit { }
class Apple : IFruit { }
class Pear : IFruit { }
class FruitBasket<TFruit> : IBasket<TFruit> where TFruit : IFruit { }
void Handle(IBasket<IFruit> what) { }
Handle(new FruitBasket<Apple>());
Handle(new FruitBasket<Pear>());
关于c# - 如何使用反射转换为通用接口(interface)?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13872359/