new InStockStatus(1424, true),
new InStockStatus(7892, false),
new InStockStatus(8534, true),
new InStockStatus(6411, true)
};
// Сформировать запрос, объединяющий объекты классов Item
//и InStockStatus для составления списка наименований товаров
// и их наличия на складе. Обратите внимание на формирование
// последовательности объектов класса Temp,
var inStockList = from item in items
join entry in statusList
on item.ItemNumber equals entry.ItemNumber
select new Temp(item.Name, entry.InStock);
Console.WriteLine("Товар\tНаличие\n");
// Выполнить запрос и вывести его результаты.
foreach(Temp t in inStockList)
Console.WriteLine("{0}\t{1}", t.Name, t.InStock);
}
}
Эта программа дает следующий результат
Товар Наличие
Кусачки True
Тиски False
Молоток True
Пила True
Для того чтобы стал понятнее принцип действия оператора join
, рассмотрим каждую строку запроса из приведенной выше программы по порядку. Этот запрос начинается, как обычно, со следующего оператора from
.
var inStockList = from item in items
В этом операторе указывается переменная диапазона item
для источника данных items
, который представляет собой массив объектов класса Item
. В классе Item
инкапсулируются наименование товара и порядковый номер товара, хранящегося на складе.
Далее следует приведенный ниже оператор join.
join entry in statusList
on item.ItemNumber equals entry.ItemNumber
В этом операторе указывается переменная диапазона entry
для источника данных statusList
, который представляет собой массив объектов класса InStockStatus
, связывающего порядковый номер товара с состоянием его запасов на складе. Следовательно, у массивов items
и statusList
имеется общее свойство: порядковый номер товара. Именно это свойство используется в части on/equals
оператора join
для описания связи, по которой из двух разных источников данных выбираются наименования товаров, когда их порядковые номера совпадают.
И наконец, оператор select
возвращает объект класса Temp
, содержащий наименование товара и состояние его запасов на складе.
select new Temp(item.Name, entry.InStock);
Таким образом, последовательность результатов, получаемая по данному запросу, состоит из объектов типа Temp
.
Рассмотренный здесь пример применения оператора join
довольно прост. Тем не менее этот оператор поддерживает и более сложные операции с источниками данных. Например, используя совместно операторы into
и join
, можно создать join
, окупаются сторицей, поскольку он дает возможность распознавать данные во время выполнения программы. Это очень ценная возможность. Но она становится еще ценнее, если используются анонимные типы, о которых речь пойдет в следующем разделе.
Анонимные типы
В C# предоставляется средство, называемое select
. Результатом запроса нередко оказывается последовательность объектов, которые составляются из членов, полученных из двух или более источников данных (как, например, в операторе join
), или же включают в себя подмножество членов из одного источника данных. Но в любом случае тип возвращаемого объекта зачастую требуется только в самом запросе и не используется в остальной части программы. Благодаря анонимному типу в подобных случаях отпадает необходимость объявлять класс, который предназначается только для хранения результата запроса.
Анонимный тип объявляется с помощью следующей общей формы:
new { имя_А = значение_А, имя_В = значение_В, ... }