Все запросы оканчиваются оператором select или group. В данном примере используется оператор select, точно определяющий, что именно должно быть по лучено по запросу. В таких простых примерах запросов, как рассматриваемый здесь, выбирается конкретное значение диапазона. Поэтому по данному запросу возвраща ются только те целые значения, которые удовлетворяют условию, указанному в опера торе where. В более сложных запросах можно дополнительно уточнять, что именно следует выбирать. Например, по запросу списка рассылки может быть получена лишь фамилия адресата вместо его полного адреса. Обратите внимание на то, что оператор select завершается точкой с запятой, поскольку это последний оператор в запросе. А другие его операторы не оканчиваются точкой с запятой.
Итак, переменная запроса posNums создана, но результаты запроса пока еще не получены. Дело в том, что сам запрос определяет лишь ряд конкретных правил, а ре зультаты будут только после выполнения запроса. Кроме того, один и тот же запрос может быть выполнен два раза или больше, причем с разными результатами, если в промежутке между последовательно производимыми попытками выполнить один и тот же запрос изменяется базовый источник данных. Поэтому одного лишь объяв ления переменной запроса posNums совершенно недостаточно для того, чтобы она со держала результаты запроса.
Для выполнения запроса в данном примере программы организуется следующий цикл. foreach(int i in posNums) Console.WriteLine(i + " ");
В этом цикле переменная posNums указывается в качестве коллекции, к которой происходит обращение на каждом шаге цикла. В цикле foreach соблюдаются прави ла, определенные в запросе и доступные по ссылке из переменной posNums. На каж дом шаге цикла возвращается очередной элемент, полученный из массива. Этот про цесс завершается, когда запрашиваемых элементов в массиве больше не обнаружено. В данном примере тип int переменной шага цикла i указывается явно, поскольку по запросу извлекаются элементы именно этого типа. Явное указание типа переменной шага цикла вполне допустимо в тех случаях, когда заранее известен тип значения, вы бираемого по запросу. Но в более сложных случаях оказывается проще, а иногда даже нужно, указывать тип переменной шага цикла неявным образом с помощью ключево го слова var. Неоднократное выполнение запросов
Итак, в запросе определяются правила, по которым извлекаются данные, но этого явно недостаточно для получения результатов, поскольку запрос должен быть выпол нен, причем это может быть сделано несколько раз. Если же в промежутке между по следовательно производимыми попытками выполнить один и тот же запрос источник данных изменяется, то получаемые результаты могут отличаться. Поэтому как только запрос определен, его выполнение будет всегда давать только самые последние резуль таты. Обратимся к конкретному примеру. Ниже приведен другой вариант рассматри ваемой здесь программы, где содержимое массива nums изменяется в промежутке между двумя последовательно производимыми попытками выполнить один и тот же запрос, хранящийся в переменной posNums. // Сформировать простой запрос. using System; using System.Linq; using System.Collections.Generic; class SimpQuery { static void Main { int[] nums = { 1, -2, 3, 0, -4, 5 }; // Сформировать простой запрос на получение только положительных значений. var posNums = from n in nums where n > 0 select n; Console.Write("Положительные значения из массива nums: "); // Выполнить запрос и отобразить его результаты. foreach(int i in posNums) Console.Write(i + " "); Console.WriteLine; // Внести изменения в массив nums. Console.WriteLine("\nЗадать значение 99 для элемента массива nums[1]."); nums[1] = 99; Consofe.Write("Положительные значения из массива nums\n" + "после изменений в нем: "); // Выполнить запрос второй раз. foreach(int i in posNums) Console.Write(i + " "); Console.WriteLine; } }
Вот к какому результату приводит выполнение этой программы. Положительные значения из массива nums: 1 3 5 Задать значение 99 для элемента массива nums[l]. Положительные значения из массива nums после изменений в нем: 1 99 3 5