简单的问题

我们在Where子句中使用了一个lambda表达式,它不返回“expected”结果.

快速总结

在analyzeObjectRepository对象中,有一些对象也包含一个名为Parent的属性中的父关系.我们正在查询此analysisObjectRepository以返回一些对象.

详情

下面的代码应该是返回根,包含ID值的特定对象的第一个孩子(直接子代)和子孙.

在下面的代码中,常识说,所有结果,使任何3个独立OR条件为真应返回,如结果.

List<AnalysisObject> analysisObjects = 
    analysisObjectRepository
        .FindAll()
        .Where(x => x.ID               == packageId ||
                    x.Parent.ID        == packageId || 
                    x.Parent.Parent.ID == packageId)
        .ToList();

但上面的代码只返回子孙,而不返回根对象(有一个null的父值),这使得

x.ID == packageId

条件是真的.

只有第二个对象

x.Parent.ID == packageId

和第三

x.Parent.Parent.ID == packageId

返回条款.

如果我们只使用下面的代码编写代码来返回根对象,那么它将被返回,所以我们完全确定analyzeObjectRepository包含所有的对象

List<AnalysisObject> analysisObjects = 
    analysisObjectRepository
        .FindAll()
        .Where(x => x.ID == packageId )
        .ToList();

但是,当我们将其重写为代理时,我们得到预期的结果,返回所有预期的对象.

List<AnalysisObject> analysisObjects = 
    analysisObjectRepository
        .FindAll()
        .Where(delegate(AnalysisObject x) 
        { 
            return 
              (x.ID == packageId) || 
              (x.Parent != null && x.Parent.ID == packageId) || 
                  (x.Parent != null && 
                   x.Parent.Parent != null && 
                   x.Parent.Parent.ID == packageId); })
        .ToList();

我们在lambda表达式中缺少一些东西吗?这是一个非常简单的3部分OR条件,我们认为任何使三个条件中的任一条件为真的对象都应该被返回.我们怀疑具有null Parent值的根对象可能会导致问题,但无法准确计算出来.

任何帮助都会很棒.

解决方法

你的第二个委托不是第一个匿名委托(而不是lambda)格式的重写.看看你的条件.

第一:

x.ID == packageId || x.Parent.ID == packageId || x.Parent.Parent.ID == packageId

第二:

(x.ID == packageId) || (x.Parent != null && x.Parent.ID == packageId) || 
(x.Parent != null && x.Parent.Parent != null && x.Parent.Parent.ID == packageId)

对lambda的调用将抛出一个异常,其中的任何一个ID不匹配,父对象为null或不匹配,并且祖父母为空.将空检查复制到lambda中,它应该正常工作.

编辑评论后问题

如果您的原始对象不是List< T>,那么我们无法知道FindAll()的返回类型,以及这是否实现了IQueryable接口.如果是这样,那么这可能解释了差异.因为lambdas可以在编译时被转换成Expression< Func>>但是匿名委托不能,那么在使用lambda版本时使用IQueryable,而在使用匿名委托版本时可能会使用LINQ对对象.

这也将解释为什么你的lambda不会导致NullReferenceException.如果要将该lambda表达式传递给实现IEnumerable< T>但是不是IQueryable< T>的lambda(与其他方法没有任何差异)的运行时间评估会在第一次遇到ID不等于目标的对象时抛出NullReferenceException,父或祖父是空值.

增加了3/16/2011 8:29 AM EDT

考虑以下简单的例子:

IQueryable<MyObject> source = ...; // some object that implements IQueryable<MyObject>

var anonymousMethod =  source.Where(delegate(MyObject o) { return o.Name == "Adam"; });    
var expressionLambda = source.Where(o => o.Name == "Adam");

这两种方法产生完全不同的结果.

第一个查询是简单的版本.匿名方法导致一个委托,然后传递给IEnumerable< MyObject>.在扩展方法中,将根据您的委托检查源的整个内容(使用普通的编译代码手动在内存中).换句话说,如果你熟悉C#中的迭代器块,那就像这样做:

public IEnumerable<MyObject> MyWhere(IEnumerable<MyObject> dataSource,Func<MyObject,bool> predicate)
{
    foreach(MyObject item in dataSource)
    {
        if(predicate(item)) yield return item;
    }
}

这里的重点是您实际上是在客户端的内存中进行过滤.例如,如果你的源是一些sql ORM,查询中就不会有WHERE子句;整个结果集将被带回客户端并在那里过滤.

使用lambda表达式的第二个查询被转换为Expression< Func< MyObject,bool>>>并使用IQueryable< MyObject> .Where()扩展方法.这导致一个也被输入为IQueryable< MyObject>的对象.所有这些都通过将表达式传递给底层提供程序.这就是为什么你没有得到NullReferenceException.查询提供程序完全取决于如何翻译表达式(而不是实际编译的函数,它可以调用,它是表示使用对象的表达式的逻辑)到它可以使用的东西.

一个简单的方法来看待区别(或至少有一个区别),就是在调用lambda版本之前调用AsEnumerable().这将强制您的代码使用LINQ对对象(这意味着它像IEnumerable< T>像匿名委托版本,而不是IQueryable< T>像lambda版本当前一样),你会得到预期的异常.

TL; DR版本

很长一段时间,你的lambda表达式被转换成对数据源的某种查询,而匿名方法版本正在评估内存中的整个数据源.无论做什么,将lambda转换成查询都不代表您期望的逻辑,这就是为什么它不会产生您期望的结果.

c# – 具有lambda表达式的LINQ where子句具有OR子句和空值返回不完整的结果的更多相关文章

  1. ios – 如何使用Magical Record创建和更新对象并保存它们而不使用contextForCurrentThread

    解决方法所以,我使用客观C而不是RubyMotion,但你应该能够这样做:编辑如果要稍后保存上下文,您只需要坚持下去:这里的主要思想是,您只需要掌握上下文并在准备就绪时执行您的操作.如果要进行后台保存,请使用以下方法:

  2. OC的Block、Swift的Closure及C#的Lambda对比

    Block:代码块、语法糖、对象Closure:闭包、自包含模块、捕获前后变量处理逻辑Lambda:可用于创建委托或表达式目录树类型的匿名函数Lambda运算符=>的左边列出了需要的参数,右边定义了赋予Lambda变量方法的实现代码。

  3. Android Studio重构还原所有lambda和其他Java 8功能

    AndroidStudioIDE中是否内置了此类功能?解决方法您可以通过将光标置于–>内来替换lambda.然后按AltEnter然后选择“用……替换lambda”您可以通过展开菜单并选择“修复所有…”来对整个文件执行此操作.您可以按照上述步骤在整个项目中执行此操作,而是单击“运行检查…”选择“整个项目”检查完成后,右键单击“Lambda可以替换…”部分并选择“将lambda替换为……”

  4. android – lambda表达式未使用

    lambda表达式如何未使用?

  5. android – kotlin lambda表达式作为可选参数

    如何将lambda表达式作为Kotlin语言中的可选参数传递我必须传递上面的东西就像可选参数解决方法下面定义了一个接受函数的函数,如果没有提供函数,则指定传递函数的默认值.您可以传递自己的功能:您可以省略该功能并使用默认值:或者,您可以将标准库maxOf函数作为默认值,而不是编写自己的函数:

  6. Python lambda 匿名函数优点和局限性深度总结

    这篇文章主要为大家介绍了Python lambda 匿名函数的优点和局限性深度总结,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

  7. 深入了解Python中Lambda函数的用法

    lambda函数是Python中常用的内置函数,又称为匿名函数。和普通函数相比,它只有函数体,省略了def和return,使得结构看起来更精简。本文将详细说说Lambda函数的用法,需要的可以参考一下

  8. Lambda表达式的使用及注意事项

    这篇文章主要介绍了Lambda表达式的使用及注意事项,主要围绕 Lambda表达式的省略模式 Lambda表达式和匿名内部类的区别的相关内容展开详情,感兴趣的小伙伴可以参考一下

  9. Java中的lambda和stream实现排序

    这篇文章主要介绍了Java中的lambda和stream实现排序,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的小伙伴可以参考一下

  10. python函数和python匿名函数lambda详解

    这篇文章主要介绍了python函数和python匿名函数lambda,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

随机推荐

  1. c# – (wpf)Application.Current.Resources vs FindResource

    所以,我正在使用C#中的WPF创建一个GUI.它看起来像这样:它现在还没有完成.这两行是我尝试制作一种数据表,它们在XAML中是硬编码的.现在,我正在C#中实现添加新的水果按钮功能.我在XAML中有以下样式来控制行的背景图像应该是什么样子:因此,在代码中,我为每列col0,col1和col2创建一个图像,如果我使用以下代码,它添加了一个如下所示的新行:如你所见,它不太正确……为什么一个似乎忽略了一些属性而另一个没有?

  2. c# – 绑定DataGridTemplateColumn

    似乎我已经打了个墙,试图在DataGrid上使用DataTemplates.我想要做的是使用一个模板来显示每个单元格的两行文本.但是似乎无法以任何方式绑定列.以下代码希望显示我想做的事情.注意每个列的绑定:模板列没有这样的东西,因此,这个xaml不可能工作.我注定要将整个DataTemplate复制到每个列,只是对每个副本都有不同的约束?解决方法我不完全确定你想要做什么,但如果您需要获取整行的DataContext,可以使用RelativeSource绑定来移动视觉树.像这样:

  3. c# – 学习设计模式的资源

    最近我来到了这个设计模式的概念,并对此感到非常热情.你能建议一些帮助我深入设计模式的资源吗?

  4. c# – 是否有支持嵌入HTML页面的跨操作系统GUI框架?

    我想开发一个桌面应用程序来使用跨系统,是否有一个GUI框架,允许我为所有3个平台编写一次代码,并具有完全可脚本化的嵌入式Web组件?我需要它有一个API来在应用程序和网页之间进行交流.我知道C#,JavaScript和一些python.解决方法Qt有这样的事情QWebView.

  5. c# – 通过字符串在对象图中查找属性

    我试图使用任意字符串访问嵌套类结构的各个部分.给出以下(设计的)类:我想要从Person对象的一个实例的“PersonsAddress.HousePhone.Number”获取对象.目前我正在使用反思来做一些简单的递归查找,但是我希望有一些忍者有更好的想法.作为参考,这里是我开发的(crappy)方法:解决方法您可以简单地使用标准的.NETDataBinder.EvalMethod,像这样:

  6. c# – 文件下载后更新页面

    FamilyID=0a391abd-25c1-4fc0-919f-b21f31ab88b7&displaylang=en&pf=true它呈现该页面,然后使用以下元刷新标签来实际向用户提供要下载的文件:你可能需要在你的应用程序中做类似的事情.但是,如果您真的有兴趣在文件完全下载后执行某些操作,那么您的运气不佳,因为没有任何事件可以与浏览器进行通信.执行此操作的唯一方法是上传附件时使用的AJAXupload.

  7. c# – 如何在每个机器应用程序中实现单个实例?

    我必须限制我的.net4WPF应用程序,以便每台机器只能运行一次.请注意,我说每个机器,而不是每个会话.我使用一个简单的互斥体实现单实例应用程序,直到现在,但不幸的是,这样一个互斥是每个会话.有没有办法创建机器互连,还是有其他解决方案来实现每个机器应用程序的单个实例?

  8. c# – WCF和多个主机头

    我的雇主网站有多个主机名,都是同一个服务器,我们只是显示不同的皮肤来进行品牌宣传.不幸的是,在这种情况下,WCF似乎不能很好地工作.我试过overridingthedefaulthostwithacustomhostfactory.这不是一个可以接受的解决方案,因为它需要从所有主机工作,而不仅仅是1.我也看过thisblogpost,但是我无法让它工作,或者不是为了解决我的问题.我得到的错误是“这

  9. c# – ASP.NET MVC模型绑定与表单元素名称中的虚线

    我一直在搜索互联网,试图找到一种方式来容纳我的表单元素的破折号到ASP.NET的控制器在MVC2,3或甚至4中的默认模型绑定行为.作为一名前端开发人员,我更喜欢在我的CSS中使用camelCase或下划线进行破折号.在我的标记中,我想要做的是这样的:在控制器中,我会传入一个C#对象,看起来像这样:有没有办法通过一些正则表达式或其他行为来扩展Controller类来适应这种情况?我讨厌这样的事实,我必须这样做:甚至这个:思考?

  10. c# – 用户界面设计工具

    我正在寻找一个用户界面设计工具来显示文档中可能的GUI.我不能生成代码.我知道MicrosoftVisio提供了一个功能.但有什么办法吗?您使用哪种软件可视化GUI?

返回
顶部