Cómo aplanar objetos anidados con expresión linq

125

Estoy tratando de aplanar objetos anidados como este:

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; }
}

Déjame hacer un ejemplo. Esta es la información que tengo

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

El resultado que estoy buscando es la siguiente lista plana:

"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"

¿Cómo podría lograr esto? Podría hacerlo con un nuevo select pero me han dicho que un SelectMany sería suficiente.

abx78
fuente

Respuestas:

199
myBooks.SelectMany(b => b.Chapters
    .SelectMany(c => c.Pages
        .Select(p => b.Name + ", " + c.Name + ", " + p.Name)));
Yuriy Faktorovich
fuente
¡¡¡Increíble!!! ¿Qué pasaría si tuviera un resultado como un nuevo objeto, como FlatBook {BookName, ChapterName, PageName}?
abx78
2
@ abx78: simplemente modifique la última selección:.Select(p => new FlatBook(b.Name, c.Name, p.Name))
user7116
¡Gracias chicos, esto era lo que necesitaba!
abx78
1
¿esto produce el mismo resultado? myBooks.SelectMany(b => b.Chapters).SelectMany(c => c.Pages).Select(p => b.Name + ", " + c.Name + ", " + p.Name);
Homero
1
@Mastro ¿qué tal?myBooks.SelectMany(b => b.Chapters == null || !b.Chapters.Any()? new []{b.Name + " has no Chapters"} : b.SelectMany(c => c.Pages.Select(p => b.Name + ", " + c.Name + ", " + p.Name)));
Yuriy Faktorovich
50

Suponiendo que bookses una lista de libros:

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};
boca
fuente
2
+1, aunque cualquiera IEnumerable<Book>lo hará, no necesita a List<Book>.
user7116
2
myBooks.SelectMany(b => b.Chapters
    .SelectMany(c => c.Pages
        .Select(p => new 
                {
                    BookName = b.Name ,
                    ChapterName = c.Name , 
                    PageName = p.Name
                });
Madhukar Bhandari
fuente
77
Si bien este ejemplo de código puede responder la pregunta, carece de explicación. Tal como está ahora, no agrega ningún valor y soporta el cambio de ser rechazado / eliminado. Agregue alguna explicación de lo que hace y por qué es una solución para el problema del OP.
2015
0

Estaba tratando de hacer esto también, y de los comentarios de Yuriy y jugando con linqPad tengo esto ...

Tenga en cuenta que no tengo libros, capítulos, páginas, tengo persona (libros), companyPerson (capítulos) y compañías (páginas)

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
                           };

o

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
         }
   )

Sitio de referencia que utilicé: http://odetocode.com/blogs/scott/archive/2008/03/25/inner-outer-lets-all-join-together-with-linq.aspx

Mastro
fuente