协变
要把泛型类型参数定义为协变,可以在类型定义中使用out
关键字,如下面的示例所示:
public interface IMethaneProducer<out T>
{
...
}
对于接口定义,协变类型参数只能用作方法的返回值或属性get
访问器。 ??
说明协变用途的一个很好例子在.NET Framework中,即前面使用的IEnumerable<T>
接口。在这个接口中,项类型T定义为协变,这表示可以把支持IEnumerable<Cow>
的对象放在IEnumerable<Animal>
类型的变量中。
因此下面的代码是有效的:
static void Main(string[] args)
{
List<Cow> cows = new List<Cow>();
cows.Add(new Cow("Gernoimo"));
cows.Add(new SuperCow("Tonto"));
ListAnimals(cows);
Console.ReadKey();
}
static void ListAnimals(IEnumerable<Animal> animals)
{
foreach(Animal animal in animals)
{
Console.WriteLine(animal.ToString());
}
}
其中cows变量的类型是List<Cow>
,它支持IEnumerable<Cow>
接口。通过协变,这个变量可以传送给需要IEnumerable<Animal>
类型的参数的方法。回顾一下froeach循环的工作方式,就知道GetEnumerator()方法用于获取IEnumerator<T>
的一个枚举器,该枚举器的Current属性用于访问项。IEnumerator<T>
还将其类型参数定义为协变,这表示可以把它用作参数的get访问器,而且一切都运转良好。