抗变

  要把泛型类型参数定义为抗变,可在类型定义中使用in关键字:

    public interface IGrassMuncher<in T>
    {
        ...
    }

  对于接口定义,抗变类型参数只能用作方法参数,不能用作返回类型。

  理解这一点的最佳方式是列举一个在.NET Framework中使用抗变的例子。带有抗变类型参数的一个接口是前面用过的IComparer<T>。可以给Animal实现这个接口,如下所示:

    public class AnimalNameLengthComparer : IComparer<Animal>
    {
        public int Compare(Animal x, Animal y)
        {
            return x.Name.Length.CompareTo(y.Name.Length);
        }
    }

  这个比较器按名称的长度比较动物,所以可以使用它对List<Animal>的实例排序。通过抗变,还可以使用它对List<Cow>的实例排序,尽管List<Cow>.Sort()方法需要IComparer<Cow>的实例。

  这个比较器按名称的长度比较动物,所以可以使用它对List<Animal>的实例排序。通过抗变,还可以使用它对List<Cow>的实例排序,尽管List<Cow>.Sort()方法需要IComparer<Cow>的实例。

    List<Cow> cows = new List<Cow>();
    cows.Add(new Cow("Geronimo"));
    cows.Add(new SuperCow("Tonto"));
    cows.Add(new Cow("Gerald"));
    cows.Add(new Cow("Phil"));
    cows.Sort(new AnimalNameLengthComparer());

  大多数情况下,抗变都会发生---它被添加到.NET Framework中就是为了帮助执行这种操作。.NET 4及更高版本中这两种变体的优点是,你可以在需要时使用本节介绍的技术实现它。

🔚