该限制的真实原因是什么?这只是必须完成的工作吗?概念上难吗?这不可能吗?
当然,人们不能在字段中使用类型参数,因为它们总是可读写的。但这不是答案,可以吗?
这个问题的原因是我在C#4上写了一篇关于方差支持的文章,我觉得我应该解释为什么它仅限于委托和接口。只是为了逆转举证责任。
更新 埃里克问了一个例子。
这个怎么样(不知道这是否有意义,但是: - ))
public class Lookup<out T> where T : Animal {
public T Find(string name) {
Animal a = _cache.FindAnimalByName(name);
return a as T;
}
}
var findReptiles = new Lookup<Reptile>();
Lookup<Animal> findAnimals = findReptiles;
在一个类中拥有它的原因可能是类本身中保存的缓存。请不要将您的不同类型的宠物命名为相同!
BTW,这让我想到optional type parameters in C# 5.0: - )
更新2:我没有声明CLR,C#应该允许这样做。只是想了解是什么原因导致它没有。
答案 0 :(得分:20)
首先,正如Tomas所说,CLR不支持它。
其次,这将如何运作?假设你有
class C<out T>
{ ... how are you planning on using T in here? ... }
T只能用于输出位置。如您所知,该类不能包含任何类型为T的字段,因为该字段可以写入。该类不能有任何采用T的方法,因为它们是逻辑写入的。假设您有此功能 - 您将如何利用它?
这对于不可变类很有用,如果我们可以说,让一个类型为T的只读字段合法;这样我们就可以大大减少不正确写入的可能性。但是很难想出其他允许以类型安全方式出现差异的场景。
如果你有这样的场景,我很乐意看到它。这将指向有朝一日在CLR中实现这一点。
更新:见
Why isn't there generic variance for classes in C# 4.0?
有关此问题的更多信息。
答案 1 :(得分:8)
据我所知,CLR不支持此功能,因此添加此功能也需要在CLR端进行大量工作。我相信在版本4.0之前CLR实际上支持接口和委托的共同和反向差异,因此这是一个相对简单的实现扩展。
(虽然支持这个类的功能绝对有用!)
答案 2 :(得分:1)
如果他们被允许,可以定义有用的100%类型安全(没有内部类型转换)类或结构,如果他们的构造函数接受一个或多个T或T供应商那么它们的类型T是协变的。可以定义有用的100%类型安全的类或结构,如果它们的构造函数接受一个或多个T消费者,则它们相对于T是逆变的。我不确定类的优于接口,除了使用“new”而不是使用静态工厂方法(很可能来自名称类似于接口的类),但我可以当然看到使用不可变结构支持协方差的用例。