В служебном классе System.Linq.Enumerable определено множество обобщенных расширяющих методов (таких как Aggregate, First, Мах<Т>() и т.д.), которые класс System.Array (и другие типы) получают в свое распоряжение на заднем плане. Таким образом, если вы примените операцию точки к локальной переменной currentVideoGames, то обнаружите большое количество членов, которые System.Array.
Роль отложенного выполнения
Еще один важный момент, касающийся выражений запросов LINQ, заключается в том, что фактически они не оцениваются до тех пор, пока не начнется итерация по результирующей последовательности. Формально это называется QueryOverlnts():
static void QueryOverInts()
{
int[] numbers = { 10, 20, 30, 40, 1, 2, 3, 8 };
// Получить числа меньше 10.
var subset = from i in numbers where i < 10 select i;
// Оператор LINQ здесь оценивается!
foreach (var i in subset)
{
Console.WriteLine("{0} < 10", i);
}
Console.WriteLine();
// Изменить некоторые данные в массиве.
numbers[0] = 4;
// Снова производится оценка!
foreach (var j in subset)
{
Console.WriteLine("{0} < 10", j);
}
Console.WriteLine();
ReflectOverQueryResults(subset);
}
На заметку! Когда оператор LINQ выбирает одиночный элемент (с использованием First()/FirstOrDefault(), Single()/SingleOrDefault() или любого метода агрегирования), запрос выполняется немедленно. Методы First(), FirstOrDefault(), Single() и SingleOrDefault будут описаны в следующем разделе. Методы агрегирования раскрываются позже в главе.
Ниже показан вывод, полученный в результате запуска программы. Обратите внимание, что во второй итерации по запрошенной последовательности появился дополнительный член, т.к. для первого элемента массива было установлено значение меньше 10:
1 < 10
2 < 10
3 < 10
8 < 10
4 < 10
1 < 10
2 < 10
3 < 10
8 < 10
Среда Visual Studio обладает одной полезной особенностью: если вы поместите точку останова перед оценкой запроса LINQ, то получите возможность просматривать содержимое во время сеанса отладки. Просто наведите курсор мыши на переменную результирующего набора LINQ (subset на рис. 13.1) и вам будет предложено выполнить запрос, развернув узел Results View (Представление результатов).
Роль немедленного выполнения
Когда требуется оценить выражение LINQ, выдающее последовательность, за пределами логики foreach, можно вызывать любое количество расширяющих методов, определенных в типе Enumerable, таких как ТоArray<Т>(), ToDictionary и ToList. Все методы приводят к выполнению запроса LINQ в момент их вызова для получения снимка данных. Затем полученным снимком данных можно манипулировать независимым образом.
Кроме того, запрос выполняется немедленно в случае поиска только одного элемента. Метод First() возвращает первый элемент последовательности (и должен всегда применяться с конструкцией orderby). Метод FirstOrDefault() возвращает стандартное значение для типа элемента в последовательности, если возвращать нечего, например, когда исходная последовательность пуста или конструкция where отбросила все элементы. Метод Single() также возвращает первый элемент последовательности (на основе orderby или согласно порядку следования элементов, если конструкция orderby отсутствует). Подобно аналогично именованному эквиваленту метод SingleOrDefault() возвращает стандартное значение для типа элемента в последовательности, если последовательность пуста (или конструкция where отбросила все элементы). Отличие между методами First(OrDefault) и Single(OrDefault) заключается в том, что Single(OrDefault) сгенерирует исключение, если из запроса будет возвращено более одного элемента.
static void ImmediateExecution()
{
Console.WriteLine();
Console.WriteLine("Immediate Execution");