不知道类型就返回通用

时间:2012-09-27 15:46:58

标签: c# generics inheritance types factory

我有一种情况,我有一个类,在其泛型类型参数中接受某个对象类型的实例。布局是这样的:

public abstract BaseClass { ... }
public DiamondClass : BaseClass { ... }
public SilverClass : BaseClass { ... }

public Handler<T> where T : BaseClass { ... }

我希望能够创建一个方法来返回Handler<DiamondClass>Handler<BaseClass>的实例,而无需在输入时定义类型。我尝试过这些方法:

public Handler<BaseClass> GetHandler(HandlerType type)
{
    switch(type)
    {
        case HandlerType.Diamond: return new Handler<DiamondClass>();
        case HandlerType.Silver: return new Handler<SilverClass>();
        default: throw new InvalidOperationException("...");
    }
}

但这不起作用,因为显然Handler<DiamondClass>不会隐含地投射到Handler<BaseClass>。我可以像这样指定:

public Handler<T> GetHandler<T>(HandlerType type) where T : BaseClass
{
    switch(type)
    {
        case HandlerType.Diamond: return (Handler<T>)new Handler<DiamondClass>();
        case HandlerType.Silver: return (Handler<T>)new Handler<SilverClass>();
        default: throw new InvalidOperationException("...");
    }
}

但现在我需要致电GetHandler<DiamondClass>GetHandler<BaseClass>。这就违背了一个方法,即在不知道类型的情况下返回基于枚举的正确处理程序的目的。我希望我可以定义一个Type对象并传递它,如下:

 Type objType = typeof(DiamondClass);
 var handler = Handler<objType>();

但显然C#不会允许那种愚蠢。我已经走了几个不同的方式,我想有办法做到这一点,但我很难过。


(实际上我确实通过返回一个dynamic对象来实现这个目的,但是如果可能的话,我想避免使用它,因为它会丢失任何类型的安全性和Intellisense支持。)

1 个答案:

答案 0 :(得分:9)

这是协方差发挥作用的地方,协方差和反方差只适用于接口和委托,因此,要解决您的问题,只需定义一个新接口IHandler as co-variant with out,指定type参数是co-variant:

public interface IHandler<out T> where T : BaseClass 
{
}

具有协变类型参数的接口使其方法能够返回比类型参数

指定的类型更多的派生类型

它会起作用。更多信息是here