我有三个继承自同一界面的clasess:
Apple:Resource
Banana:Resource
Orange:Resource
我在另一个类中列出了每个类型为op的列表:
private List<Apple> apples = new List<Banana>();
private List<Banana> bananas = new List<Empleado>();
private List<Orange> oranges = new List<Orange>();
现在我想要一个可以返回任何列表的方法,让我们这样说:
public List<Resource> getListOf(string resource)
{
switch (resource)
{
case "apple":
return apples;
break;
case "bananas":
return bananas;
break;
case "oranges":
return oranges;
break;
default:
Console.WriteLine("wrong fruit");
break;
}
}
这给了我以下错误:
无法隐式转换类型 '
System.Collections.Generic.List<Fruits.Apple>
'到 'System.Collections.Generic.List<Fruits.Resource>
'
我认为它的工作方式与我能做的相同:
foreach (Resource r in Fruitlist) {
r.Meth();
}
答案 0 :(得分:7)
使代码因缺席而中断的功能称为泛型类型协方差。 C#确实支持variance in generics,但只支持接口(不是具体类)。
foreach
代码段有效,因为它将Fruitlist
视为IEnumerable<Fruit>
,因为这是带有out
(协变)类型参数的通用接口,编译器可以查看它作为IEnumerable<Resource>
。
如果getListOf
可以接受IEnumerable<Resource>
,则更改函数签名将使代码编译。否则编译器将无法保证类型安全,因为您可以编写此代码:
List<Resource> apples = this.getListOf("apple");
apples.Add(new Orange()); // adding a Resource to a List<Resource>, right?
另一种选择是返回非泛型类型,在这种情况下,类型安全性将再次丢失,编译器将引入运行时检查以确保您不会将苹果与橙子混合(但至少代码会如果表现得很好就行。)