dev-notes/docs/languages/dotnet/csharp/linq.md

85 lines
3.9 KiB
Markdown
Raw Normal View History

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
```cs
// 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.
```cs
// 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>`
```cs
Enumerable.Range(int start, int end); // IEnumerable<int> of values between start & end
IEnumerable<TSource>.Select(Func<TSource, TResult> selector); // map
IEnumerable<TSource>.Where(Func<T, bool> predicate); // filter
2021-01-31 11:05:37 +01: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>
IEnumerable<T>.Max();
2021-02-26 09:21:20 +01:00
IEnumerable<T>.Min();
2021-01-31 11:05:37 +01: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
2021-02-26 09:21:20 +01:00
```cs
Enumerable.Method(IEnumerable<T> source, args);
// if extension method same as
IEnumerable<T>.Method(args);
2021-01-31 11:05:37 +01:00
```