与LINQ有些混淆

Ste*_*ane 0 .net c# linq linq-to-objects

一些背景信息;

  • LanguageResource是基类
  • LanguageTranslatorResource和LanguageEditorResource继承自LanguageResource
  • LanguageEditorResource定义了一个IsDirty属性
  • LanguageResourceCollection是LanguageResource的集合
  • LanguageResourceCollection在内部保存LanguageResources Dictionary<string, LanguageResource> _dict
  • LanguageResourceCollection.GetEnumerator()返回 _dict.Values.GetEnumerator()

我有一个LanguageResourceCollection _resources只包含LanguageEditorResource对象,并希望使用LINQ来枚举那些脏的,所以我尝试了以下内容.我的具体问题是粗体.

  1. _resources.Where(r => (r as LanguageEditorResource).IsDirty)

    但是,Intellisense没有显示其他LINQ方法,但我仍然对它进行编码,并告诉我"LanguageResourceCollection不包含'Where'的定义,也没有扩展方法......".

    为什么LanguageResourceCollection实现IEnumerable的方式阻止它支持LINQ?

  2. 如果我将查询更改为

    (_resources as IEnumerable<LanguageEditorResource>).Where(r => r.IsDirty)

    Intellisense,则显示LINQ方法和解决方案.但是在运行时我得到一个ArgumentNullException"Value不能为null.参数名称:source".

    这是我的LINQ代码中的问题吗?
    这是一般类设计的问题吗?
    我如何深入了解LINQ生成的内容以尝试查看问题所在?

我对这个问题的目的不是为特定问题找到解决方案,因为我现在必须使用其他(非LINQ)方法解决它,而是尝试提高我对LINQ的理解并学习如何改进设计我的课程与LINQ更好地合作.

Mar*_*ell 6

这听起来像你的集合实现IEnumerable,而不是IEnumerable<T>,因此你需要:

_resources.Cast<LanguageEditorResource>().Where(r => r.IsDirty)
Run Code Online (Sandbox Code Playgroud)

请注意,它Enumerable.Where是定义的IEnumerable<T>,而不是IEnumerable- 如果您具有非泛型类型,则需要使用Cast<T>(或OfType<T>)来获取正确的类型.不同之处在于,Cast<T>如果它发现某个不是a的东西会抛出异常T,其中 - OfType<T>只是忽略任何不是a的东西T.由于您已经声明您的集合包含LanguageEditorResource,因此使用Cast<T>而不是静默删除数据来检查该假设是合理的.

还要检查您是否"使用System.Linq"(并引用System.Core(.NET 3.5;以及带有.NET 2.0的LINQBridge)来获取Where扩展方法.

实际上,使用你的集合实现是值得的IEnumerable<LanguageResource>- 你可以简单地使用Cast<T>方法或迭代器块(yield return).

[编辑]建立在理查德普尔的笔记上 - 你可以在这里编写自己的通用容器,大概是T : LanguageResource(并T在其中使用Dictionary<string,T>,并实现IEnumerable<T>ICollection<T>).只是一个想法.