4

我对 LINQ 还很陌生,我可能已经把自己画到了这里的一个角落。我有两个列表(左和右),我需要:

a ) 根据特定字段获取匹配项

b ) 获取左侧的项目,右侧没有匹配项

c ) 获取右侧的项目,左侧​​没有匹配项

如果某些字段相等,则找到匹配项。其他字段可能包含也可能不包含值,但不得影响匹配比较。

为了获得项目aJOIN在两个列表上都执行了 a :

var q = from a in r1
        from b in r2
        where a.Prop1 == b.Prop1 && a.Prop3 == b.Prop3
        select new { a.Prop1, a.Prop2, b.Prop3 };

我不确定从这里去哪里。我我不能使用.Except(),因为两个列表的其他属性会有所不同,并且可能会导致比较中断。

我还尝试使用Left Join并获取不匹配的项目:

 var q =
     from c in r1
     join p in r2 on c.Prop1 equals p.Prop1
     into cp
     from p in cp.DefaultIfEmpty()
     select new { Prop1 = c.Prop1, Prop2 = p == null ? "N/A" : p.Prop2 };

但是我发现您不能比较多个要比较的字段。

Left JoinLINQ上可以有多个字段吗?还有其他方法(除了 LINQ)来获得两个列表之间的差异吗?

4

3 回答 3

4

这使用IntersectExcept(类似于 Cuong Le 的解决方案):

public class MyComparer : IEqualityComparer<YourClass>
{
    #region IEqualityComparer<YourClass> Members

    public bool Equals(YourClass x, YourClass y)
    {
        return
            x.Prop1.Equals(y.Prop1) && x.Prop3.Equals(y.Prop3);
    }

    public int GetHashCode(YourClass obj)
    {
        int hCode = obj.Prop1.GetHashCode() ^ obj.Prop3.GetHashCode();
        return hCode.GetHashCode();
    }

    #endregion
}

// matched elements from both lists
var r1 = l1.Intersect<YourClass>(l2, new MyComparer());
// elements from l1 not in l2
var r2 = l1.Except<YourClass>(l2, new MyComparer());
// elements from l2 not in l1
var r3 = l2.Except<YourClass>(l1, new MyComparer());
于 2012-09-26T09:45:15.473 回答
2

默认情况下,Except方法使用EqualityComparer.Default,这就是为什么如果您有具有不同属性值的对象时不能使用

但是你可以使用另一个重载方法除了自定义 EqualityComparer<T>,假设它忽略Pro3

public class CustomComparer : EqualityComparer<A>
{
    public override int GetHashCode(A a)
    {
        int hCode = a.Pro1.GetHashCode() ^ a.Pro2.GetHashCode();
        return hCode.GetHashCode();
    }

    public override bool Equals(A a1, A a2)
    {
        return a1.Pro1.Equals(a2.Pro1) && a1.Pro2.Equals(a2.Pro2)
    }
}

然后你可以使用Except

listA.Except(listB, new CustomComparer());
于 2012-09-26T09:41:53.560 回答
0

怎么样

一种)

r1.Where(x=>r2.Any(y=>x.Prop1==y.Prop1&&x.Prop3==y.Prop3))
    .Select(x=>new { x.Prop1,x.Prop2,x.Prop3});

b)

r1.Where(x=>!r2.Any(y=>x.Prop1==y.Prop1&&x.Prop3==y.Prop3))
    .Select(x=>new { x.Prop1,x.Prop2,x.Prop3});

C)

r2.Where(x=>!r1.Any(y=>x.Prop1==y.Prop1&&x.Prop3==y.Prop3))
    .Select(x=>new { x.Prop1,x.Prop2,x.Prop3});
于 2012-09-26T09:41:36.897 回答