行为
假设您有两个列表:
Id Value1 A2 B3 CId ChildValue1 a11 a21 a32 b12 b2
当您 Join
在该
Id字段上的两个列表时,结果将是:
Value ChildValueA a1A a2A a3B b1B b2
当您 GroupJoin
在该
Id字段上的两个列表时,结果将是:
Value ChildValuesA [a1, a2, a3]B [b1, b2]C []
因此,
Join产生父值和子值的平坦(表格)结果。
GroupJoin在第一个列表中产生一个条目列表,每个条目在第二个列表中都有一组联接的条目。
这就是在SQL中
Join与之等效的原因
INNER JOIN:没有的条目
C。While 与结果集中
GroupJoin的
OUTERJOIN:等效
C,但相关条目的列表为空(在SQL结果集中将存在一行
C - null)。
句法
因此,让两个列表分别为
IEnumerable<Parent>和
IEnumerable<Child>。(对于Linq to Entities
:)
IQueryable<T>。
Join
语法将是
from p in Parentjoin c in Child on p.Id equals c.Idselect new { p.Value, c.ChildValue }返回
IEnumerable<X>where
X是具有两个属性的匿名类型,
Value和
ChildValue。此查询语法使用幕后
Join方法。
GroupJoin
语法将是
from p in Parentjoin c in Child on p.Id equals c.Id into gselect new { Parent = p, Children = g }返回
IEnumerable<Y>where
Y是一个匿名类型,该匿名类型由type的一个属性和type的
Parent一个属性组成
IEnumerable<Child>。此查询语法使用幕后
GroupJoin方法。
我们可以
selectg在后面的查询中进行选择,这将选择一个
IEnumerable<IEnumerable<Child>>,例如一个列表列表。在许多情况下,包含父项的选择更为有用。
一些用例
1.产生平坦的外部连接。
如前所述,该声明…
from p in Parentjoin c in Child on p.Id equals c.Id into gselect new { Parent = p, Children = g }…产生一个有孩子群体的父母名单。可以通过两个小的添加将其转换为平面的父子对列表:
from p in parentsjoin c in children on p.Id equals c.Id into g // <= intofrom c in g.DefaultIfEmpty() // <= flattens the groupsselect new { Parent = p.Value, Child = c?.ChildValue }结果类似于
Value ChildA a1A a2A a3B b1B b2C (null)
注意,在上面的语句中重新使用了 范围变量
c。这样做,
join只要将
outer join等价物添加
into g from c ing.DefaultIfEmpty()到现有
join语句中,就可以简单地将其转换为。
这是查询(或综合)语法的亮点。方法(或流利的)语法显示了实际发生的情况,但是很难编写:
parents.GroupJoin(children, p => p.Id, c => c.Id, (p, c) => new { p, c }) .SelectMany(x => x.c.DefaultIfEmpty(), (x,c) => new { x.p.Value, c?.ChildValue } )因此
outer joinLINQ中的单位是
GroupJoin,由展平
SelectMany。
2.保存顺序
假设父母的名单更长一些。某些用户界面会
Id以固定顺序将所选父母的列表作为值生成。让我们使用:
var ids = new[] { 3,7,2,4 };现在,必须按照正确的顺序从父母列表中过滤出选定的父母。
如果我们做…
var result = parents.Where(p => ids.Contains(p.Id));
…的顺序
parents将决定结果。如果按排序
Id,则结果将是父母2、3、4、7。不好。但是,我们也可以使用
join过滤列表。并通过使用
ids第一个列表,将保留顺序:
from id in idsjoin p in parents on id equals p.Idselect p
结果是父母3、7、2、4。



