Тип переменной диапазона должен соответствовать типу элементов, хранящихся в источнике данных. Следовательно, тип переменной диапазона зависит от типа источника данных. Как правило, тип переменной диапазона может быть выведен средствами С#. Но выводимость типов может быть осуществлена при условии, что в источнике данных реализована форма интерфейса IEnumerable
, где Т обозначает тип элементов в источнике данных. (Как упоминалось выше, форма интерфейса IEnumerable
реализуется во всех массивах, как, впрочем, и во многих других источниках данных.) Но если в источнике данных реализован необобщенный вариант интерфейса IEnumerable
, то тип переменной диапазона придется указывать явно. И это делается в операторе from
. Ниже приведен пример явного объявления типа int
переменной диапазона n.
var posNums = from int n in nums // ...
Очевидно, что явное указание типа здесь не требуется, поскольку все массивы неявно преобразуются в форму интерфейса IEnumerable
, которая позволяет вывести тип переменной диапазона.
Тип объекта, возвращаемого по запросу, представляет собой экземпляр интерфейса IEnumerable
, где Т — тип получаемых элементов. Следовательно, тип переменной запроса должен быть экземпляром интерфейса IEnumerable
, а значение Т должно определяться типом значения, указываемым в операторе select
. В предыдущих примерах значению Т соответствовал тип int
, поскольку переменная n имела тип int. (Как пояснялось выше, переменная n относится к типу int
, потому что элементы именно этого типа хранятся в массиве nums
.) С учетом явного указания типа IEnumerable
упомянутый выше запрос можно было бы составить следующим образом.
IEnumerable
where n > 0 select n;
Следует иметь в виду, что тип элемента, выбираемого оператором select
, должен соответствовать типу аргумента, передаваемого форме интерфейса IEnumerable
, используемой для объявления переменной запроса. Зачастую при объявлении переменных запроса используется ключевое слово var
вместо явного указания ее типа, поскольку это дает компилятору возможность самому вывести соответствующий тип данной переменной из оператора select
. Как будет показано далее в этой главе, такой подход оказывается особенно удобным в тех случаях, когда оператор select
возвращает из источника данных нечто более существенное, чем отдельный элемент.
Когда запрос выполняется в цикле foreach
, тип переменной шага цикла должен быть таким же, как и тип переменной диапазона. В предыдущих примерах тип этой переменной указывался явно как int
. Но имеется и другая возможность: предоставить компилятору самому вывести тип данной переменной, и для этого достаточно указать ее тип как var
. Как будет показано далее в этой главе, ключевое слово var приходится использовать и в тех случаях, когда тип данных просто неизвестен.
Общая форма запроса
У всех запросов имеется общая форма, основывающаяся на ряде приведенных ниже контекстно-зависимых ключевых слов.
Ascending
by
descending equals
from
group
in
into
join
let
on
orderby
select
where
Среди них лишь приведенные ниже ключевые слова используются в начале операторов запроса.
from
group
join
let
orderby
select
where
Запрос должен начинаться с ключевого слова from
и оканчиваться ключевым словом select
или group
. Оператор select
определяет тип значения, перечисляемого по запросу, а оператор group
возвращает данные группами, причем каждая группа может перечисляться по отдельности. Как следует из приведенных выше примеров, в операторе where
указываются критерии, которым должен удовлетворять искомый элемент, чтобы быть полученным по запросу. А остальные операторы позволяют уточнить запрос. Все они рассматриваются далее по порядку.
Отбор запрашиваемых значений с помощью оператора where
Как пояснялось выше, оператор where служит для отбора данных, возвращаемых по запросу. В предыдущих примерах этот оператор был продемонстрирован в своей простейшей форме, в которой для отбора данных используется единственное условие. Однако для более тщательного отбора данных можно задать несколько условий и, в частности, в нескольких операторах where
. В качестве примера рассмотрим следующую программу, в которой из массива выводятся только те значения, которые положительны и меньше 10.
// Использовать несколько операторов where.
using System;
using System.Linq;
class TwoWheres {
static void Main() {
int [ ] nums = { 1, -2, 3, -3, 0, -8, 12, 19, 6, 9, 10 };