将两个查询的左外连接转换为 LINQ

Posted

技术标签:

【中文标题】将两个查询的左外连接转换为 LINQ【英文标题】:Converting left outer join on two queries to LINQ 【发布时间】:2015-09-07 10:59:41 【问题描述】:

我有下面的 sql 查询,我想将其转换为 LINQ 以获得完全相同的结果并由下面的查询返回

    select *
             from (
                select distinct DocID 
                from UserViewDoc 
                where UserViewDoc.UVID in (102558)) a 
         left outer join
            (
                select distinct UserViewDoc.DocID 
                from UserViewDoc 
                     inner join UserViewHeader on UserViewDoc.UVID =        UserViewHeader.UVID 
                where UserViewDoc.UVID not in (102558) 
                  and UserViewHeader.IsLock = 1) b on a.DocID = b.DocID
         where b.DocID is  null
         )

到目前为止我尝试过的都在 LINQ 语句下面

   var v = (from uvd in this.ViewSelectorControl.LDReviewContext.GetTable<UserViewDoc>()
                 where IDs.Contains(uvd.UVID)
                 select new  uvd.DocID, uvd.UVID );
        var c = ((from uvd in this.ViewSelectorControl.LDReviewContext.GetTable<UserViewDoc>()
                 join uvh in this.ViewSelectorControl.LDReviewContext.GetTable<UserViewHeader>() on uvd.UVID equals uvh.UVID
                 where !IDs.Contains(uvh.UVID) && uvh.IsLock == true
                 select new  uvd.DocID, uvd.UVID  ));
        var d = (from id in v
                 join ids in c on id.UVID equals ids.UVID into vc
                 from sub in vc.DefaultIfEmpty()
                 where sub == null
                 select id);

我面临的问题是运行 SQL 查询返回 30583 条记录,它的 LINQ 版本返回所有 30613 条记录

【问题讨论】:

很高兴您展示了自己的努力,但您能否说明为什么它显然没有达到您的目标?此外,通常可以通过使用导航属性而不是 LINQ 连接来解决连接问题。 @GertArnold 谢谢!!,对不起,我错过了那个,实际上当我运行 sql 查询时它返回 30583 条记录,但在 LINQ 中运行相同的东西会返回所有 30613 条记录。另外,我可以使用 Navigational 属性,因为这最终会将一百万条记录加载到内存中 您的 SQL 和您的 LINQ 语句不会做同样的事情。这可能解释了返回行的差异。您的实际 SQL 是什么? 能否请您提供一些 sql 代码,这些代码将创建表并允许在表中插入一些初始值以了解您的代码的作用? @YuraZaletskyy 我不能给出实际的表,但假设两个表都包含 UVID 和 DocID,并且表 UserViewHeader 包含更多字段 IsLock。现在,我想要的是来自 UserViewDoc 的所有记录,除了那些 IsLock 在特定 UVID 中为真的记录 【参考方案1】:

我把查询改写成和sql查询一模一样,我相信是这样的:

var firstQuery = (from u in this.ViewSelectorControl.LDReviewContext.GetTable<UserViewDoc>()
                  where IDs.Contains(u.UVID)
                  select u.DocID).Distinct();

var innerQuery = (from u in this.ViewSelectorControl.LDReviewContext.GetTable<UserViewDoc>()
                      join uvh in this.ViewSelectorControl.LDReviewContext.GetTable<UserViewHeader>() on uvh.UVID equals u.UVID
                      where IDs.Contains(u.UIVD) == false
                      && uvh.IsLock
                      select u.DocID).Distinct();

var resultQuery = from f in firstQuery
                  join i in innerQuery on i equals f into lout
                  from i in lout.DefaultIfEmpty()
                  where i == null
                  select f;

【讨论】:

以上是关于将两个查询的左外连接转换为 LINQ的主要内容,如果未能解决你的问题,请参考以下文章

LINQ查询中的左外连接[重复]

如何在 Linq 中执行左外连接? [复制]

LINQ:具有多个条件的左外连接

linq中的左外连接

使用 Linq to Sql 的左外连接结果问题

左外连接和多重计数 SQL to LINQ