Lamda 表达式里的Join和GroupJoin的区别, 如何实现SQL的Left Join效果

摘要:
例如,您可以将产品表与产品类别表连接,以获取产品名称及其对应的类别名称db.Products.Join.Where;该连接对应于SQL中的InnerJoin,这很容易理解。上述代码中红色c的类型为类别。如果我想在SQL中实现LeftJoin的效果,我应该如何编写它?

例如,可以将产品表与产品类别表相联接,得到产品名称和与其相对应的类别名称

db.Products  
.Join  
 (  
      db.Categories,  
      p => p.CategoryID,  
      c => c.CategoryID,  
      (p, c) => new {p,c.CategoryName}  
)  
.Where(p => p.CategoryID == 1);  

这个Join对应SQL里的inner Join, 这个很容易理解. 上面代码里红色的c的类型是Category

如果我要达到SQL里Left Join的效果, 该怎么写呢? 想当然的以为,把Join改成GroupJoin,

db.Products  
.GroupJoin  
 (  
      db.Categories,  
      p => p.CategoryID,  
      c => c.CategoryID,  
      (p, cs) => new {p,cs.FirstOrDefault().CategoryName} 
)  
.Where(p => p.CategoryID == 1);  

上面代码里红色的cs的类型是IEnumerable<Category>, 你用FirstOrDefault()只是拿到一个目录,万一Product Left join是有多个目录呢?

所以要用.SelectMany把IEnumerable<Category> 把里面的每一个元素做投影.

要注意加上DefaultIfEmtyp() 才是Left join, 假如没有加这个,还是Inner Join

db.Products
.GroupJoin(db.Categories,
          (Product p) => p.CategoryId,
          (Category c) => c.CategoryId,
          (prod, cs) => new { prod, cs }) // cs is IEnumerable<Category>
          .SelectMany(prodCats => prodCats.cs.DefaultIfEmpty(), (prodCats, c) =>
                                new
                                {
                                    prodCats.prod
                                    categoryName = c.CategoryName
                                });

用LinQ query表达式 对应的写法是

            var qry =
            from p in db.Products
            join c in db.Categories on  p.CategoryId equals c.CategoryId
            into g from pc in g.DefaultIfEmpty()
            select new { 
                    prodId = p.ProdId,
                    prodCode = p.ProdCode,
                    prodName = p.ProdName,
                    categoryName = g.FirstOrDefault().CategoryName
            };

如果我只用DefaultIfEmpty(),不用GroupJoin,这样写能达到这个效果吗?

            var qry = db.Products.SelectMany
            (
                p => db.Categories.Where(c => c.CategoryId == p.CategoryId).DefaultIfEmpty(),
                (p, c) => new
                {
                    p.ProdId,
                    p.ProdCode,
                    p.ProdName,
                    c.CategoryName
                }
            );

这样写,如果数据量大,性能会很差. 因为它会把Products和Categories都Load到内存里,再Match, 而不是用SQL Left Join

参考文章: 

https://stackoverflow.com/questions/584820/how-do-you-perform-a-left-outer-join-using-linq-extension-methods

http://linqsamples.com/linq-to-objects/join/GroupJoin-linq

https://docs.microsoft.com/en-us/dotnet/csharp/linq/perform-left-outer-joins 

免责声明:文章转载自《Lamda 表达式里的Join和GroupJoin的区别, 如何实现SQL的Left Join效果》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇filter()数组遍历WPF省市联动Binding下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

随便看看

基于 WebRTC 的 RTSP 视频实时预览

该方案采用基于WebRTC的视频即时消息,其原生支持RTP协议的解码,因此延迟可以非常低,约为0.2-0.4秒。其他方案的延迟大于1秒。WebRTC需要浏览器。您可以在以下地址查看支持的浏览器。WebRTC实现基于web的视频会议。标准是WHATWG协议。其目的是通过浏览器提供简单的javascript来实现实时通信功能。Github中有很多WebRTC的实...

js 预览 excel,js-xlsx的使用

js-xlsx简介SheetJS生成的js-xls x是一个非常方便的工具库,只能使用纯js读取和导出excel。它功能强大,支持多种格式,支持xls、xlsx和ods等十几种格式。本文以xlsx格式为例。官方github:https://github.com/SheetJS/js-xlsx支持演示在线演示地址:http://demo.haoji.me/20...

Vue浏览器调试工具VueTools安装以及使用

ue-devtools是一款基于chrome浏览器的插件,用于vue应用的调试,这款vue调试神器可以极大地提高我们的调试效率。vue-devtools使用起来还是比较简单的,上手非常的容易,这里就细讲其使用说明了。安装方法二:这里以chrome浏览器为例:1、打开chrome网上应用店,搜索vue.js注:如果打不开页面需要代理选择第一个,点击添加至chr...

2022年可用QQ机器人框架

4.小李子机器人官网:https://xiaolz.cn评估:支持多个Q登录和论坛似乎是目前最活跃的。它支持许多api,可以满足许多需求。没有限制,但有很多错误。...

JRebel激活服务搭建

前言因为平时的开发工具是使用IntelliJIDEA,所以热部署项目代码的时候,使用的Jrebel。因为Jrebel是收费的,所以以前用的时候都是在网上找破解方法,在网上找到的办法是输入一个在线激活服务,来进行激活。由于简单方便就一直这样用的,今天早上打开IDEA后发现,Jrebel激活失效了。JRebel很好用,也是离不开大家的支持,所以如果条件允许的话,...

The server selected protocol version TLS10 is not accepted by client preferences [TLS12]

修改jre中的这个文件:/java/jdk1.8/jre/lib/security/java.securityjdk.tls.legacyAlgorithms直接改成下面的这个:jdk.tls.legacyAlgorithms=SSLv3,RC4,DES,MD5withRSA,DHkeySize˂1024,ECkeySize˂224,anon,NULL,in...