5

我正在使用 C#.NET 3.5。我明白了

参数类型“GenericTest.BarkStrategy”不可分配给参数类型“GenericsTest.IAnimalStrategy”

使用以下(对于这个问题尽可能简化)代码:

using System.Collections.Generic;

namespace GenericsTest {
    class Program {
        static void Main(string[] args) {
            List<IAnimalStrategy<IAnimal>> strategies = 
                new List<IAnimalStrategy<IAnimal>>();
            strategies.Add(new BarkStrategy());
        }
    }

    interface IAnimal { }
    interface IAnimalStrategy<T> where T : IAnimal { }
    class Dog : IAnimal { }
    class BarkStrategy : IAnimalStrategy<Dog> { }
}
4

1 回答 1

14

你必须告诉编译器你的接口是协变的:IAnimalStrategy<out T>

namespace GenericsTest
{
    class Program
    {
        static void Main(string[] args)
        {
            List<IAnimalStrategy<IAnimal>> strategies = new List<IAnimalStrategy<IAnimal>>();
            strategies.Add(new BarkStrategy());
        }
    }

    interface IAnimal { }
    interface IAnimalStrategy<out T> where T : IAnimal { }

    class Dog : IAnimal { }
    class BarkStrategy : IAnimalStrategy<Dog> { }
}

不幸的是,它仅在 C# 4.0 中可用:How is Generic Covariance & Contra-variance Implemented in C# 4.0?

为了理解这个问题,你可以忘记这个列表,这一行不编译:

IAnimalStrategy<IAnimal> s = new BarkStrategy();

接口可以在IAnimalStrategy<IAnimal>IAnimal 上做事,也许设置一个类型的属性IAnimal

interface IAnimalStrategy<T> where T : IAnimal 
{
    T Animal {get; set;}
}

然后你就可以做类似的事情

IAnimalStrategy<IAnimal> s = new BarkStrategy();
s.Animal = new Cat();

它会吹到你的脸上。所以 C#3.5 不允许你这样做。
如果您说 T 与out关键字协变,C#4.0 将允许您这样做

interface IAnimalStrategy<out T> where T : IAnimal 
{
    T Animal {get; set;}
}

这又要吹了

无效方差:类型参数“T”必须在 IAnimalStrategy.Animal 上始终有效。“T”是协变的。

协变和逆变很难理解,我建议你阅读 Eric Lippert 博客上的精彩系列:C# 中的协变和逆变

于 2013-10-23T12:35:04.740 回答