C#如何为通用子代迭代常见父类的词典?

时间:2019-03-18 04:56:40

标签: c# covariance contravariance

我有这种情况:

class Animal 
{
  int size;
} 

class Dog : Animal 
{
 string Name;
}

class Cat : Animal 
{
 string Alias;
}

public void Check( Dictionary<string,Animal> animals ) 
{
 foreach(var pet in animals){...}
}

我的问题在这里:

如果狗是动物,为什么大公国不承认狗为动物?

您是否有想法允许传递子类的Dictionary并对其进行迭代(不使用(string,object)的对待Dictionary)?

public void MySituation()

    {
        Dictionary<string,Animal> dogs = new Dictionary<string,Dog>();
        ** ERROR 01**

        Dictionary<string,Cat> cats = new Dictionary<string,Cat>();
        Check(cats);
        ** ERROR 02**
    }

非常感谢。 编译器的优化能力与您同在。

1 个答案:

答案 0 :(得分:1)

Dictionary<string,Animal>的问题在于它允许您添加。如果您可以在“狗”词典中添加“动物”,那么您很容易在其中养猫,那么地狱就会崩溃。因此,编译器不允许您将任何其他类型的字典视为相同。

您需要做的是使用一个只读接口,例如IEnumerable

public void Check( IEnumerable<KeyValuePair<string,Animal>> animals ) 
{
     foreach(var pair in animals)
     {
         var animal = pair.Value;
         //Do something
     }
}

另一个选择是使您的方法通用:

public void Check<T>( Dictionary<string,T>> animals ) where T : animal
{
     foreach(T animal in animals)
     {
         //Do something
     }
}

如果您想了解其背后的理论,请参阅有关协方差和逆方差的答案之一,例如this onethis one