Полное руководство. С# 4.0
Шрифт:
Эта программа дает следующий результат. Квадратные корни положительных значений, округленные до двух десятичных цифр: 4.05 3.48 10.04 5.02
Обратите особое внимание в данном примере запроса на следующий оператор select. select Math.Sqrt(n);
Он возвращает квадратный корень значения переменной диапазона. Для этого зна чение переменной диапазона передается методу Math.Sqrt, который возвращает квадратный корень своего аргумента. Это означает, что последовательность результа тов, получаемых при выполнении запроса, будет содержать квадратные корни поло жительных значений, хранящихся в массиве nums. Если обобщить этот принцип, то его эффективность станет вполне очевидной. Так, с помощью оператора select мож но сформировать любой требующийся тип последовательности результатов, исходя из значений, получаемых из источника данных.
Ниже приведена программа, демонстрирующая другое применение оператора select. В этой программе сначала создается
Вот к какому результату приводит выполнение этой программы. Адреса электронной почты: Herb@HerbSchildt.com Tom@HerbSchildt.com Sara@HerbSchildt.com
Обратите особое внимание на следующий оператор select. select entry.Address;
Вместо полного значения переменной диапазона этот оператор возвращает лишь его адресную часть (Address). Это означает, что по данному запросу возвращается последовательность символьных строк, а не объектов типа EmailAddress. Именно поэтому переменная s указывается в цикле foreach как string. Ведь как пояснялось ранее, тип последовательности результатов, возвращаемых по запросу, определяется типом значения, возвращаемым оператором select.
Одной из самых эффективных для оператора select является возможность возвра щать последовательность результатов, содержащую элементы данных, формируемые во время выполнения запроса. В качестве примера рассмотрим еще одну программу. В ней определяется класс ContactInfo, в котором хранится имя, адрес электронной почты и номер телефона адресата. Кроме того, в этой программе определяется класс EmailAddress, использовавшийся в предыдущем примере. В методе Main создает ся массив объектов типа ContactInfo, а затем объявляется запрос, в котором источ ником данных служит этот массив, но возвращаемая последовательность результатов содержит объекты типа EmailAddress. Таким образом, типом последовательности результатов, возвращаемой оператором select, является класс EmailAddress, а не класс ContactInfo, причем его объекты создаются во время выполнения запроса. // Использовать запрос для получения последовательности объектов // типа EmailAddresses из списка объектов типа ContactInfo. using System; using System.Linq; class ContactInfo { public string Name { get; set; } public string Email { get; set; } public string Phone { get; set; } public ContactInfo(string n, string a, string p) { Name = n; Email = a; Phone = p; } } class EmailAddress { public string Name { get; set; } public string Address { get; set; } public EmailAddress(string n, string a) { Name = n; Address = a; } } class SelectDemo3 { static void Main { ContactInfo[] contacts = { new Contactlnfo("Герберт", "Herb@HerbSchildt.com", "555-1010"), new Contactlnfo("Tom", "Tom@HerbSchildt.com", "555-1101"), new Contactlnfo("Capa", "Sara@HerbSchildt.com", "555-0110") }; // Сформировать запрос на получение списка объектов типа EmailAddress. var emailList = from entry in contacts select new EmailAddress(entry.Name, entry.Email); Console.WriteLine("Список адресов электронной почты:"); // Выполнить запрос и вывести его результаты. foreach(EmailAddress е in emailList) Console.WriteLine(" {0}: {1}", e.Name, e.Address ); } }
Ниже приведен результат выполнения этой программы. Список адресов электронной почты: Герберт: Herb@HerbSchildt.com Том: Tom@HerbSchildt.com Сара: Sara@HerbSchildt.com
Обратите особое внимание в данном запросе на следующий оператор select. select new EmailAddress(entry.Name, entry.Email);
В этом операторе создается новый объект типа EmailAddress, содержащий имя и адрес электронной почты, получаемые из объекта типа ContactInfo, хранящегося в массиве contacts. Но самое главное, что новые объекты типа EmailAddress созда ются в операторе select во время выполнения запроса. Применение вложенных операторов from
Запрос может состоять из нескольких операторов from, которые оказываются в этом случае вложенными. Такие операторы from находят применение в тех случаях, когда по запросу требуется получить данные из двух разных источников. Рассмотрим простой пример, в котором два вложенных оператора from используются в запросе для циклического обращения к элементам двух разных массивов символов. В итоге по такому запросу формируется последовательность результатов, содержащая все воз можные комбинации двух наборов символов. // Использовать два вложенных оператора from для составления списка // всех возможных сочетаний букв А, В и С с буквами X, Y и Z. using System; using System.Linq; // Этот класс содержит результат запроса. class ChrPair { public char First; public char Second; public ChrPair(char c, char c2) { First = c; Second = c2; } } class MultipleFroms { static void Main { char[] chrs = { 'A', 'B', 'C' }; char[] chrs2 = { 'X', 'Y', 'Z' }; // В первом операторе from организуется циклическое обращение // к массиву символов chrs, а во втором операторе from — // циклическое обращение к массиву символов chrs2. var pairs = from ch1 in chrs from ch2 in chrs2 select new ChrPair(ch1, ch2); Console.WriteLine("Все сочетания букв ABC и XYZ: "); foreach(var p in pairs) Console.WriteLine("{0} {1}", p.First, p.Second); } }
Выполнение этого кода приводит к следующему результату. Все сочетания букв ABC и XYZ: А X A Y A Z В X В Y В Z С X С Y С Z
Этот пример кода начинается с создания класса ChrPair, в котором содержатся результаты запроса. Затем в нем создаются два массива, chrs и chrs2, и, наконец, формируется следующий запрос для получения всех возможных комбинаций двух по следовательностей результатов. var pairs = from ch1 in chrs from ch2 in chrs2 select new ChrPair(ch1, ch2);
Во вложенных операторах from организуется циклическое обращение к обоим массивам символов, chrs и chrs2. Сначала из массива chrs получается символ, со храняемый в переменной ch1. Затем перечисляется содержимое массива chrs2. На каждом шаге циклического обращения во внутреннем операторе from символ из массива chrs2 сохраняется в переменной сh2 и далее выполняется оператор select. В результате выполнения оператора select создается новый объект типа ChrPair, содержащий пару символов, которые сохраняются в переменных ch1 и ch2 на каждом шаге циклического обращения к массиву во внутреннем операторе from. А в конечном итоге получается объект типа ChrPair, содержащий все возможные сочетания извле каемых символов.
Вложенные операторы from применяются также для циклического обращения к источнику данных, который содержится в другом источнике данных. Соответствую щий пример приведен в разделе "Применение оператора let для создания времен ной переменной в запросе" далее в этой главе. Группирование результатов с помощью оператора group
Одним из самых эффективных средств формирования запроса является оператор group, поскольку он позволяет группировать полученные результаты по ключам. Ис пользуя последовательность сгруппированных результатов, можно без особого тру да получить доступ ко всем данным, связанным с ключом. Благодаря этому свойству оператора group доступ к данным, организованным в последовательности связанных элементов, осуществляется просто и эффективно. Оператор group является одним из двух операторов, которыми может оканчиваться запрос. (Вторым оператором, завер шающим запрос, является select.) Ниже приведена общая форма оператора group. group переменная_диапазона by ключ
Этот оператор возвращает данные, сгруппированные в последовательности, при чем каждую последовательность обозначает общий ключ.
Результатом выполнения оператора group является последовательность, состоя щая из элементов типа IGrouping<TKey, TElement>, т.е. обобщенного интерфейса, объявляемого в пространстве имен System.Linq. В этом интерфейсе определена кол лекция объектов с общим ключом. Типом переменной запроса, возвращающего груп пу, является IEnumerable<IGrouping<TKey, TElement>>. В интерфейсе IGrouping определено также доступное только для чтения свойство Key, возвращающее ключ, связанный с каждой коллекцией.
Ниже приведен пример, демонстрирующий применение оператора group. В коде этого примера сначала объявляется массив, содержащий список веб-сайтов, а затем формируется запрос, в котором этот список группируется по имени домена самого верхнего уровня, например .org или .com. // Продемонстрировать применение оператора group. using System; using System.Linq; class GroupDemo { static void Main { string[] websites = { "hsNameA.com", "hsNameB.net", "hsNameC.net", "hsNameD.com", "hsNameE.org", "hsNameF.org", "hsNameG.tv", "hsNameH.net", "hsNameI.tv" }; // Сформировать запрос на получение списка веб-сайтов, // группируемых по имени домена самого верхнего уровня. var webAddrs = from addr in websites where addr.LastIndexOf('.') != -1 group addr by addr.Substring(addr.LastIndexOf('.')); // Выполнить запрос и вывести его результаты. foreach(var sites in webAddrs) { Console.WriteLine("Веб-сайты, сгруппированные " + "по имени домена" + sites.Key); foreach(var site in sites) Console.WriteLine(" " + site); Console.WriteLine; } } }