2021-01-31 11:05:37 +01:00
# LINQ
## LINQ to Objects
<!-- Page: 423/761 of "Ian Griffiths - Programming C# 8.0 - Build Cloud, Web, and Desktop Applications.pdf" -->
The term **LINQ to Objects** refers to the use of LINQ queries with any `IEnumerable` or `IEnumerable<T>` collection directly, without the use of an intermediate LINQ provider or API such as [LINQ to SQL ](https://docs.microsoft.com/en-us/dotnet/framework/data/adonet/sql/linq/ ) or [LINQ to XML ](https://docs.microsoft.com/en-us/dotnet/standard/linq/linq-xml-overview ).
LINQ to Objects will be used when any `IEnumerable<T>` is specified as the source, unless a more specialized provider is available.
### Query Expressions
2021-02-26 09:21:20 +01:00
All query expressions are required to begin with a `from` clause, which specifies the source of the query.
2021-09-20 19:35:32 +02:00
The final part of the query is a `select` (or `group` ) clause. This determines the final output of the query and its system type.
2021-01-31 11:05:37 +01:00
2024-06-16 19:14:59 +02:00
```cs
2021-01-31 11:05:37 +01:00
// query expression
var result = from item in enumerable select item;
// where clause
var result = from item in enumerable where condition select item;
// ordering
2021-09-20 19:35:32 +02:00
var result = from item in enumerable orderby item.property select item; // ordered IEnumerable
2021-01-31 11:05:37 +01:00
2021-09-20 19:35:32 +02:00
// let clause, assign expression to variable to avoid re-evaluation on each cycle
var result = from item in enumerable let tmp = < sub-expr > ... // BEWARE: compiled code has a lot of overhead to satisfy let clause
2021-01-31 11:05:37 +01:00
// grouping (difficult to re-implement to obtain better performance)
var result = from item in enumerable group item by item.property; // returns IEnumerable< IGrouping < TKey , TElement > >
```
### How Query Expressions Expand
2021-02-26 09:21:20 +01:00
The compiler converts all query expressions into one or more method calls. Once it has done that, the LINQ provider is selected through exactly the same mechanisms that C# uses for any other method call.
2021-01-31 11:05:37 +01:00
The compiler does not have any built-in concept of what constitutes a LINQ provider.
2024-06-16 19:14:59 +02:00
```cs
2021-01-31 11:05:37 +01:00
// expanded query expression
var result = Enumerable.Where(item => condition).Select(item => item);
```
The `Where` and `Select` methods are examples of LINQ operators. A LINQ operator is nothing more than a method that conforms to one of the standard patterns.
### Methods on `Enumerable` or `IEnumerable<T>`
2024-06-16 19:14:59 +02:00
```cs
2021-01-31 11:05:37 +01:00
Enumerable.Range(int start, int end); // IEnumerable< int > of values between start & end
2021-10-04 22:37:01 +02:00
IEnumerable< TSource > .Select(Func< TSource , TResult > selector); // map
IEnumerable< TSource > .Where(Func< T , bool > predicate); // filter
2021-01-31 11:05:37 +01:00
2021-10-04 22:37:01 +02:00
IEnumerable< T > .FirstOrDefault(); // first element of IEnumerable or default(T) if empty
IEnumerable< T > .FirstOrDefault(T default); // specify returned default
IEnumerable< T > .FirstOrDefault(Func< T , bool > predicate); // first element to match predicate or default(T)
// same for LastOrDefault & SingleOrDefault
IEnumerable< T > .Chunk(size); // chunk an enumerable into slices of a fixed size
2021-01-31 11:05:37 +01:00
2021-02-26 09:21:20 +01:00
// T must implement IComparable< T >
2021-10-04 22:37:01 +02:00
IEnumerable< T > .Max();
2021-02-26 09:21:20 +01:00
IEnumerable< T > .Min();
2021-01-31 11:05:37 +01:00
2021-10-04 22:37:01 +02:00
// allow finding maximal or minimal elements using a key selector
IEnumerable< TSource > .MaxBy(Func< TSource , TResult > selector);
IEnumerable< TSource > .MinBy(Func< TSource , TResult > selector);
2021-02-26 09:21:20 +01:00
IEnumerable< T > .All(Func< T , bool > predicate); // check if condition is true for all elements
IEnumerable< T > .Any(Func< T , bool > predicate); // check if condition is true for at least one element
2021-03-04 21:59:30 +01:00
IEnumerable< T > .Concat(IEnumerable< T > enumerable);
// Applies a specified function to the corresponding elements of two sequences, producing a sequence of the results.
2021-09-20 19:35:32 +02:00
IEnumerable< TFirst > .Zip(IEnumerable< TSecond > enumerable, Func< TFirst , TSecond , TResult > func);
2021-03-04 21:59:30 +01:00
IEnumerable< TFirst > .Zip(IEnumerable< TSecond > enumerable); // Produces a sequence of tuples with elements from the two specified sequences.
2021-02-26 09:21:20 +01:00
```
2021-01-31 11:05:37 +01:00
2022-08-06 10:48:24 +02:00
> **Note**: `Enumerable` provides a set of `static` methods for querying objects that implement `IEnumerable<T>`. Most methods are extensions of `IEnumerable<T>`
2021-01-31 11:05:37 +01:00
2024-06-16 19:14:59 +02:00
```cs
2021-02-26 09:21:20 +01:00
Enumerable.Method(IEnumerable< T > source, args);
// if extension method same as
2021-10-04 22:37:01 +02:00
IEnumerable< T > .Method(args);
2021-01-31 11:05:37 +01:00
```