Code Review Stack Exchange is a question and answer site for peer programmer code reviews. It's 100% free, no registration required.

Sign up
Here's how it works:
  1. Anybody can ask a question
  2. Anybody can answer
  3. The best answers are voted up and rise to the top

A project I'm working on often has lists within lists within lists etc. The original author often uses complicated linq expressions, but when adapting them I mostly get hopelessly bogged down and resort to foreach's which makes me feel like a lesser being (joke). With an expression such as the following, what would the equivalent Linq expression be, and would you bother taking the time to make it, instead of the 'easy' foreach option.

        var participantList = new List<string>();

        foreach (var community in communities)
        {
            foreach (var tournament in community.Tournaments)
            {
                foreach (var round in tournament.Rounds)
                {
                    foreach (var match in round.Matches)
                    {
                        if (match.Home.ParticipantId.IsNotNull())
                        {
                           participantList.Add(match.Home.ParticipantId); 
                        }
                        if (match.Away.ParticipantId.IsNotNull())
                        {
                           participantList.Add(match.Away.ParticipantId); 
                        }
                    }

                }
            }
        }
share|improve this question
1  
The desire to improve code is implied for all questions on this site. Question titles should reflect the purpose of the code, not how you wish to have it reworked. See How to Ask. – Alien Herb Nite 8 hours ago
up vote 5 down vote accepted

The LINQ would be:

var participantList = communities.SelectMany(c => c.Tournaments
                                                   .SelectMany(t => t.Rounds
                                                                     .SelectMany(r => r.Matches)))
                                  .Where(m => m.Home.ParticipantId.IsNotNull() ||
                                              m.Away.ParticipantId.IsNotNull())
                                  .ToList();

LINQ does not add much imo, if the logic was more complicated the for loops are nicer to debug.

One downside with LINQ for this is that it requires formatting to be readable. If you rename things the formatting needs to be maintained.

With the foreach loops you get formatting for free.

Edit: As per @RobH's suggestion:

var participantList = communities.SelectMany(c => c.Tournaments)
                                 .SelectMany(t => t.Rounds)
                                 .SelectMany(r => r.Matches)
                                 .Where(m => m.Home.ParticipantId.IsNotNull() || 
                                             m.Away.ParticipantId.IsNotNull())
                                 .ToList();

Chaining is indeed nicer.

share|improve this answer
    
perfect, thanks! – BMills 7 hours ago
1  
+1 You could 'stack' the SelectMany calls instead of nesting them. That can be a bit easier to read. – RobH 7 hours ago

Your Answer

 
discard

By posting your answer, you agree to the privacy policy and terms of service.

Not the answer you're looking for? Browse other questions tagged or ask your own question.