3

参考在 Visual Studio 2010 express 中为 C# 编译的这个测试代码

public class Test
{
    class Base { }
    class Derived : Base { }

    void Test1(IEnumerable<Derived> derived)
    {
        IEnumerable<Base> b = derived; //This works fine using covariance on IEnumerable
    }

    void Test2<TDerived, TBase>(TDerived derived) where TDerived : TBase
    {
        TBase b = derived; //This works fine because TDerived is constrained to derive from TBase
    }

    void Test3<TDerived, TBase>(IEnumerable<TDerived> derived) where TDerived : TBase
    {
        IEnumerable<TBase> b = derived; //ERROR: paraphrased: Cannot implicitly convert type IEnumerable<TDerived> to IEnumerable<TBase>
    }
}

我试图利用 IEnumerable 的协方差将泛型类型参数的可枚举存储在该类型参数被限制继承的类的可枚举中。这以 Test3 为例。请注意,Test1 和 Test2(分别演示编译时类型的协方差和约束类型的分配)都可以正常编译。这是对我不起作用的两种语言功能的组合。

如果我的理解没有缺陷,我可以使用IEnumerable<TBase> b = derived.Cast<TBase>()并且 100% 确信任何演员都不会失败,所以我确实有一个可用的解决方法。我的问题是,为什么编译器不允许这样做?这是出于某种逻辑原因,编译器中的疏忽还是我没有想到的其他原因而被禁止?

4

1 回答 1

11

回答最初的问题

您当前正在尝试将 type 的单个元素转换为typeTDerived序列Base。我也不希望你的Cast电话能奏效,因为TDerived没有实现IEnumerable——我怀疑你实际上已经让它在不同的情况下工作了。

我怀疑你实际上的意思是:

void Test3<TDerived>(IEnumerable<TDerived> derived) where TDerived : Base
{
    IEnumerable<Base> b = derived;
}

编译没有问题。

回答已编辑的问题

好的,现在我们已经解决了两个类型参数之间的真正问题,问题是编译器不知道它们是引用类型——这是泛型变化所必需的。您可以通过以下class约束来解决此问题TDerived

void Test3<TDerived, TBase>(IEnumerable<TDerived> derived)
    where TDerived : class, TBase
{
    IEnumerable<TBase> b = derived;
}
于 2013-08-24T10:41:10.940 回答