在LINQ to Objects中预过滤连接的元素

时间:2015-05-04 21:22:01

标签: c# linq filter

我有两个集合,'left'和'right',具有不同的元素类型。元素类型都有一个字符串属性'paramNumber',它不是唯一的,并且通过它的关系是多对多的。现在,有两个DateTime字段,'left.date'和'right.startDate'。对于每个选中的'left.date',我需要得到'right'中的最后一个元素,其中'right.startDate'小或相等(除了连接条件'left.paramNumber'=='right.paramNumber')。

我想知道是否有没有完全交叉连接的直接方法?我尝试使用子查询,但遇到了'lefty'变量的范围问题(参见代码)。

我看到网上的教程首先进行完全交叉连接,然后在'where'子句中删除不需要的行,但这对我们来说是没有选择的。

class left
{
    public string paramNumber;
    public DateTime date;
    public string leftyName;
}

class right
{
    public string paramNumber;
    public DateTime startDate;
    public string anotherString;
    public DateTime timeOfDay;
}

class Program
{
    static void Main(string[] args)
    {

        List<left> lefts = new List<left>();
        List <right> rights = new List <right>();

        //lefty not visible in the where clause...
        //trying it with a second 'from right...' did not work because of
        //casting problems.
        var query =
            from left lefty in lefts
            join right righty in
               ((from right rightTmp in rights
                 where ((rightTmp.paramNumber == lefty.paramNumber) &&
                        (rightTmp.startDate <= lefty.date) &&
                        (rightTmp.anotherString == "N") )
                 select rightTmp
                ).ToList().Last())
            on lefty.paramNumber equals righty.paramNumber
            select new
            {
                myDate = lefty.date,
                myLeftyName = lefty.leftyName,
                myParamNumber = lefty.paramNumber,
                myTimeOfDay = righty.timeOfDay
            };
    }
}

[编辑]:解决方案(基于cechode的回答)

以下是我目前使用的解决方案,基于cechode的回答。 'let'子句是主要的缺失部分,因为它也允许引用l / lefty,而不仅仅是r / rightTmp。

最后,我对其中的一个所需元素进行了完全过滤 '权利'直接在第一个'let'子句中(通过升序和使用排序) 持续();当然cechode的降序/ First()也可以工作)并且不用第二个 使用'toprightcheck'变量(我认为cechode有很好的理由进行空检查,但我会尝试使用较短的版本):

var X = (from l in lefts
         let topright =
            (from r in rights
             where r.paramNumber == l.paramNumber && 
                   r.startDate <= l.date
             orderby r.startDate ascending
             select r).Last()        
         select new {
            lName = l.paramNumber,
            rname = topright.paramNumber,
            ldate = l.date,
            rdate = topright.startDate });

1 个答案:

答案 0 :(得分:1)

这对你有用吗?

var X = (from l in lefts
        let topright = (from r in rights where r.paramNumber == l.paramNumber && r.startDate < l.date orderby r.startDate descending select r)
        where topright!=null
        let toprightcheck = topright.First()
     select new { lName = l.paramNumber, rname = toprightcheck.paramNumber, ldate = l.date, rdate = toprightcheck.startDate });