执行左外部联接

左外部联接是这样定义的:返回第一个集合的每个元素,无论该元素在第二个集合中是否有任何相关元素。 可以使用 LINQ 通过对分组联接的结果调用 DefaultIfEmpty 方法来执行左外部联接。

备注

本主题中的示例使用执行内部联接中的 PetPerson 数据类。

示例

下面的示例演示如何对分组联接的结果调用 DefaultIfEmpty 方法来执行左外部联接。

若要生成两个集合的左外部联接,第一步是使用分组联接执行内联。 (有关此过程的说明,请参阅执行内联。)在此示例中,Person 对象列表基于与 Pet.Owner 匹配的 Person 对象内联到 Pet 对象列表。

第二步是在结果集内包含第一个(左)集合的每个元素,即使该元素在右集合中没有匹配的元素也是如此。 这是通过对分组联接中的每个匹配元素序列调用 DefaultIfEmpty 来实现的。 此示例中,对每个匹配 Pet 对象的序列调用 DefaultIfEmpty。 如果对于任何 Person 对象,匹配 Pet 对象序列均为空,则该方法返回一个包含单个默认值的集合,从而确保结果集合中显示每个 Person 对象。

注意

引用类型的默认值为 null;因此,该示例在访问每个 Pet 集合的每个元素之前会先检查是否存在空引用。

Person magnus = new("Magnus", "Hedlund");
Person terry = new("Terry", "Adams");
Person charlotte = new("Charlotte", "Weiss");
Person arlene = new("Arlene", "Huff");

Pet barley = new("Barley", terry);
Pet boots = new("Boots", terry);
Pet whiskers = new("Whiskers", charlotte);
Pet bluemoon = new("Blue Moon", terry);
Pet daisy = new("Daisy", magnus);

// Create two lists.
List<Person> people = new() { magnus, terry, charlotte, arlene };
List<Pet> pets = new() { barley, boots, whiskers, bluemoon, daisy };

var query =
    from person in people
    join pet in pets on person equals pet.Owner into gj
    from subpet in gj.DefaultIfEmpty()
    select new
    {
        person.FirstName,
        PetName = subpet?.Name ?? string.Empty
    };

foreach (var v in query)
{
    Console.WriteLine($"{v.FirstName + ":",-15}{v.PetName}");
}

record class Person(string FirstName, string LastName);
record class Pet(string Name, Person Owner);

// This code produces the following output:
//
// Magnus:        Daisy
// Terry:         Barley
// Terry:         Boots
// Terry:         Blue Moon
// Charlotte:     Whiskers
// Arlene:

请参阅