Why the order of LINQ to objects methods counts

前端 未结 5 2099
一整个雨季
一整个雨季 2021-02-14 15:58

I read this question\'s answers that explain the order of the LINQ to objects methods makes a difference. My question is why?

If I write a LINQ to SQL query, it doesn\'t

相关标签:
5条回答
  • 2021-02-14 16:26

    Because, with LINQ for SQL, the SQL grammar for SELECT mandates that the different clauses occur in a particular sequence. The compiler must generate grammatically correct SQL.

    Applying LINQ for objects on an IEnumerable involves iterating over the IEnumerable and applying a sequence of actions to each object in the IEnumerable. Order matters: some actions may transform the object (or the stream of objects itself), others may throw objects away (or inject new objects into the stream).

    The compiler can't divine your intent. It builds code that does what you said to do in the order in which you said to do it.

    0 讨论(0)
  • 2021-02-14 16:31

    Linq to Objects does not reorder to avoid a would-be run-time step to do something that should be optimized at coding time. The resharpers of the world may at some point introduce code analysis tools to smoke out optimization opportunities like this, but it is definitely not a job for the runtime.

    0 讨论(0)
  • 2021-02-14 16:45

    Why it doesn't work this way with LINQ to objects?

    LINQ to Objects doesn't use expression trees. The statement is directly turned into a series of method calls, each of which runs as a normal C# method.

    As such, the following in LINQ to Objects:

       var results = collection.OrderBy(x => x.Id)
                       .Where(x => x.Name == "gdoron")
                       .ToList();
    

    Gets turned into direct method calls:

       var results = Enumerable.ToList(
                       Enumerable.Where(
                         Enumerable.OrderBy(collection, x => x.Id),
                         x => x.Name = "gdoron"
                       )
                     );
    

    By looking at the method calls, you can see why ordering matters. In this case, by placing OrderBy first, you're effectively nesting it into the inner-most method call. This means the entire collection will get ordered when the resutls are enumerated. If you were to switch the order:

       var results = collection
                       .Where(x => x.Name == "gdoron")
                       .OrderBy(x => x.Id)
                       .ToList();
    

    Then the resulting method chain switches to:

       var results = Enumerable.ToList(
                       Enumerable.OrderBy(
                         Enumerable.Where(collection, x => x.Name = "gdoron"),
                         x => x.Id
                       )
                     );
    

    This, in turn, means that only the filtered results will need to be sorted as OrderBy executes.

    0 讨论(0)
  • 2021-02-14 16:45

    Linq to objects's deferred execution works differently than linq-to-sql's (and EF's).

    With linq-to-objects, the method chain will be executed in the order that the methods are listed—it doesn't use expression trees to store and translate the whole thing.

    Calling OrderBy then Where with linq-to-objects will, when you enumerate the results, sort the collection, then filter it. Conversely, filtering results with a call to Where before sorting it with OrderBy will, when you enumerate, first filter, then sort. As a result the latter case can make a massive difference, since you'd potentially be sorting many fewer items.

    0 讨论(0)
  • 2021-02-14 16:50

    It's perfectly legal to use side-effecting operations. Compare:

    "crabapple"
        .OrderBy(c => { Console.Write(c); return c; })
        .Where(c => { Console.Write(c); return c > 'c'; })
        .Count();
    "crabapple"
        .Where(c => { Console.Write(c); return c > 'c'; })
        .OrderBy(c => { Console.Write(c); return c; })
        .Count();
    
    0 讨论(0)
提交回复
热议问题