如何使用 linq 表达式展平嵌套对象

Posted

技术标签:

【中文标题】如何使用 linq 表达式展平嵌套对象【英文标题】:How to flatten nested objects with linq expression 【发布时间】:2011-09-19 17:06:42 【问题描述】:

我正在尝试像这样展平嵌套对象:

public class Book

    public string Name  get; set; 
    public IList<Chapter> Chapters  get; set; 


public class Chapter

    public string Name  get; set; 
    public IList<Page> Pages  get; set; 



public class Page

    public string Name  get; set; 

让我举个例子。这是我的数据

Book: Pro Linq 
 
   Chapter 1: Hello Linq 
   
      Page 1, 
      Page 2, 
      Page 3
   ,
   Chapter 2: C# Language enhancements
   
      Page 4
   ,

我正在寻找的结果是以下平面列表:

"Pro Linq", "Hello Linq", "Page 1"
"Pro Linq", "Hello Linq", "Page 2"
"Pro Linq", "Hello Linq", "Page 3"
"Pro Linq", "C# Language enhancements", "Page 4"

我怎么能做到这一点?我可以使用 select new 来做到这一点,但有人告诉我 SelectMany 就足够了。

【问题讨论】:

【参考方案1】:

假设books 是书籍列表:

var r = from b in books
    from c in b.Chapters
    from p in c.Pages
    select new BookName = b.Name, ChapterName = c.Name, PageName = p.Name;

【讨论】:

+1,即使任何IEnumerable&lt;Book&gt; 都可以,但不需要List&lt;Book&gt;【参考方案2】:
myBooks.SelectMany(b => b.Chapters
    .SelectMany(c => c.Pages
        .Select(p => new 
                
                    BookName = b.Name ,
                    ChapterName = c.Name , 
                    PageName = p.Name
                );

【讨论】:

虽然这个代码示例可以回答这个问题,但它缺乏解释。就目前而言,它没有增加任何价值,并且承受着被否决/删除的变化。请添加一些解释是什么以及为什么它是 OP 问题的解决方案。【参考方案3】:

我也在尝试这样做,并且从 Yuriy 的 cmets 和 linqPad 弄乱了我有这个..

请注意,我没有书籍、章节、页面,我有人员(书籍)、公司人员(章节)和公司(页面)

from person in Person
                           join companyPerson in CompanyPerson on person.Id equals companyPerson.PersonId into companyPersonGroups
                           from companyPerson in companyPersonGroups.DefaultIfEmpty()
                           select new
                           
                               ContactPerson = person,
                               ContactCompany = companyPerson.Company
                           ;

Person
   .GroupJoin (
      CompanyPerson, 
      person => person.Id, 
      companyPerson => companyPerson.PersonId, 
      (person, companyPersonGroups) => 
         new  
         
            person = person, 
            companyPersonGroups = companyPersonGroups
         
   )
   .SelectMany (
      temp0 => temp0.companyPersonGroups.DefaultIfEmpty (), 
      (temp0, companyPerson) => 
         new  
         
            ContactPerson = temp0.person, 
            ContactCompany = companyPerson.Company
         
   )

我使用的参考网站:http://odetocode.com/blogs/scott/archive/2008/03/25/inner-outer-lets-all-join-together-with-linq.aspx

【讨论】:

【参考方案4】:
myBooks.SelectMany(b => b.Chapters
    .SelectMany(c => c.Pages
        .Select(p => b.Name + ", " + c.Name + ", " + p.Name)));

【讨论】:

太棒了!!!如果我有一个新对象的结果,比如 FlatBookBookName, ChapterName, PageName 怎么办? @abx78:只需更改最后一个选择:.Select(p =&gt; new FlatBook(b.Name, c.Name, p.Name)) 谢谢你们,这正是我需要的! 这会产生相同的结果吗? myBooks.SelectMany(b =&gt; b.Chapters).SelectMany(c =&gt; c.Pages).Select(p =&gt; b.Name + ", " + c.Name + ", " + p.Name); @Mastro 怎么样myBooks.SelectMany(b =&gt; b.Chapters == null || !b.Chapters.Any()? new []b.Name + " has no Chapters" : b.SelectMany(c =&gt; c.Pages.Select(p =&gt; b.Name + ", " + c.Name + ", " + p.Name)));

以上是关于如何使用 linq 表达式展平嵌套对象的主要内容,如果未能解决你的问题,请参考以下文章

如何使用 linq 查询具有嵌套列表的对象?

如何使用两个列表推导展平嵌套的字符串列表

如何使用 pyspark 在 aws 胶水中展平嵌套 json 中的数组?

如何使用 Java 手动展平 Elasticsearch 嵌套的 JSON 文档?

如何展平嵌套的结果?

如何使用嵌套字典列表展平熊猫数据框中的列