首页
学习
活动
专区
工具
TVP
发布
社区首页 >问答首页 >在TElement中,ILookup<TKey,TElement>不应该是(声明的)协变吗?

在TElement中,ILookup<TKey,TElement>不应该是(声明的)协变吗?
EN

Stack Overflow用户
提问于 2013-02-15 04:46:16
回答 1查看 588关注 0票数 19

定义System.Linq.ILookUp为

代码语言:javascript
复制
interface ILookup<TKey, TElement> : IEnumerable<IGrouping<TKey, TElement>>, IEnumerable {
    int Count {
        get;
    }

    IEnumerable<TElement> this[TKey key] {
        get;
    }

    bool Contains(TKey key);
}

由于IEnumerable在IGrouping、IGrouping在TElement中是协变的,并且接口只公开TElement作为返回类型,所以我假设ILookup在TElement中也是协变的。实际上,这个定义

代码语言:javascript
复制
interface IMyLookup<TKey, out TElement> : IEnumerable<IGrouping<TKey, TElement>>, IEnumerable {
    int Count {
        get;
    }

    IEnumerable<TElement> this[TKey key] {
        get;
    }

    bool Contains(TKey key);
}

编译时没有问题。

那么,原始定义中缺少out关键字的原因可能是什么呢?它是否会被添加到Linq的未来版本中?

EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2013-02-21 17:57:21

追踪MSDN文档,Covariance and Contravariance in Generics已经在.NET Framework4中引入。在此之前,从.NET Framework2.0到.NET Framework3.5都有IEnumerable<T>。然后,在.NET Framework4.0中,我们可以看到类型参数T作为协方差的IEnumerable<out T>

IGrouping<TKey, TElement>ILookup<TKey, TElement>从.NET Framework3.5开始就存在了。在.NET Framework4.0中,前者已更新为IGrouping<out TKey, out TElement>,但后者已被省略,未指明原因。

TKey不能是协变的,因为Contains(TKey)this[TKey]的实现阻止了这种情况。

关于TElement,这个问题还不清楚。我不相信设计师们只是错过了它。也许原因在于对未来的计划。或者他们想阻止类似下面这样的事情,但我不知道为什么:

代码语言:javascript
复制
string[] strings = new[] {"a", "a", "b", "b", "b", "c"};
ILookup<string, string> lookup = strings.ToLookup(s => s); // Valid.
ILookup<string, object> lookup = strings.ToLookup(s => s); // Now invalid, but would correct if TElement was covariant (out TElement).

也有其他作者关注这个问题:

ToLookup

有一点奇怪,那就是IGrouping在TKey和TElement中是协变的,而ILookup在两个类型参数中都是不变的。虽然TKey必须是不变的,但对于TElement来说,协变

是合理的

票数 7
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/14883727

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档