2012-12-14 131 views
2

我有一本字典,我正在使用它来促进一些基于api版本号的内部路由。本质上发生的是我在字典中查找和操作,并尝试将其称为RUN方法。但为了做到这一点,我需要能够将对象投射到它的界面上。这里是我的意思是,这是字典:如何使用反射投射到通用界面?

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); 

然而,接口是通用的,因为它需要知道什么Request和Response类型:

var myInstance = (MyInterface<TRequest, TResponse>) Activator.CreateInstance(myOperation); 

我不知道如何告诉它这些请求和响应类型在这个阶段。我假设它可以用反射来完成。我发现我可以通过类似myOperation.GetGenericArguments()的方式获得这些泛型参数,但我不确定在这个阶段如何使用这些参数。有谁知道如何做到这一点?

回答

4

这本质上是不可能的。
在编译时你不能表达这种类型,因为直到运行时才知道类型参数。
因此,您不能投射到它或具有该类型的变量。

如果可能,您应该使接口协变,并将其强制转换为使用基本类型。

或者,您可以制作非通用或协变基础接口并使用它。

+0

你能举个例子吗?我不确定我以前做过这样的事情。 – Sinaesthetic

+0

你可以在这里了解所有关于协变接口的知识:[http://msdn.microsoft.com/en-us/library/dd997386.aspx]其实质是你可以用类型的'out'关键字来定义你的接口,意味着该接口可以接受派生较少的类而不会出错。尽管这是一个棘手的领域,但如果你不注意如何设计接口,可能会引入一些有趣的错误。 –

4

为了在SLaks扩大回答:

有处理您的方案不符合逻辑的方式。你要做的是在运行时使用不同类型的代码。这只能通过建立2个独立的代码分支(可以完成)或回退到动态/反射来完成。为了澄清这一点:

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); 
} 

现在我们可以声明一个基本类型为苹果和梨两个,即水果SLaks建议坚持。这样,Handle可以接受一个水果,并对苹果和梨上的常用功能执行逻辑。

这引发了一个问题,如何用泛型做到这一点。泛型默认情况下不支持差异,但在.NET 4.0中它肯定是可能的。您可以通过在type参数上应用out关键字来声明一个接口(但只有一个接口)为协变。这可以让你做这样的事情:

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>()); 
+0

简明扼要但内容翔实的答案。 – ean5533