Как сгладить вложенные объекты с помощью выражения linq

95

Я пытаюсь сгладить вложенные объекты следующим образом:

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"

Как я мог это сделать? Я мог бы сделать это с помощью нового, но мне сказали, что SelectMany будет достаточно.

Теги:
linq

4 ответа

158
Лучший ответ
myBooks.SelectMany(b => b.Chapters
    .SelectMany(c => c.Pages
        .Select(p => b.Name + ", " + c.Name + ", " + p.Name)));
  • 0
    Потрясающие!!! Что если у меня получится новый объект, например FlatBook {BookName, ChapterName, PageName}?
  • 2
    @ abx78: просто измените последний выбор:. Выберите .Select(p => new FlatBook(b.Name, c.Name, p.Name))
Показать ещё 15 комментариев
42

Предполагая, что 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
    +1, даже если подойдет любой IEnumerable<Book> , ему не нужен List<Book> .
  • 0
    Это хороший момент. Спасибо
Показать ещё 1 комментарий
1
myBooks.SelectMany(b => b.Chapters
    .SelectMany(c => c.Pages
        .Select(p => new 
                {
                    BookName = b.Name ,
                    ChapterName = c.Name , 
                    PageName = p.Name
                });
  • 5
    Хотя этот пример кода может ответить на вопрос, в нем нет объяснения. Как он стоит сейчас, он не добавляет никакой ценности и выдерживает изменение, которое было понижено / удалено. Пожалуйста, добавьте некоторые объяснения, что делает и почему это решение проблемы ОП.
0

Я тоже пытался это сделать, и от комментариев Юрия и возиться с 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

Ещё вопросы

Сообщество Overcoder
Наверх
Меню