Асинхронное удаленное взаимодействие
Асинхронное удаленное взаимодействие
В завершение нашего обсуждения материала данной главы давайте выясним, как вызывать члены удаленного типа асинхронно. В главе 14 была рассмотрена тема асинхронного вызова методов с помощью типов делегата. Как и следует ожидать, при асинхронном вызове удаленного объекта компоновочным блоком клиента первым шагом должно быть определение пользовательского делегата, представляющего соответствующий удаленный метод. После этого вызывающая сторона для вызова метода и получения возвращаемых значений может использовать любой из подходов, описанных в главе 14.
Для примера создайте новое консольное приложение (AsyncWKOCarProvider-Client) и установите в нем ссылку на первый вариант компоновочного блока CarGeneralAsm.dll. Теперь измените класс Program так, как показано ниже:
class Program {
// Делегат для метода GetAllAutos().
internal delegate List‹JamesBondCar› GetAllAutosDelegate();
static void Main(string[] args) {
Console.WriteLine("Старт клиента! Для завершения нажмите ‹Enter›");
RemotingConfiguration.Configure("AsyncWKOCarProviderClient.exe.config");
// Создание поставщика машин.
CarProvider cp = new CarProvider();
// Создание делегата.
GetAllAutosDelegate getCarsDel = new GetAllAutosDelegate(cp.GetAllAutos);
// Асинхронный вызов GetAllAutos().
IAsyncResult ar = getCarsDel.BeginInvoke(null, null);
// Имитация активности клиента.
while (!ar.IsCompleted) { Console.WriteLine("Клиент работает…"); }
// Все сделано! Получение возвращаемого значения делегата.
List‹JamesBondCar allJBCs = getCarsDel.EndInvoke(ar);
// Использование всех машин из списка.
foreach(JamesBondCar j in allJBCs) UseCar(j);
Console.ReadLine();
}
}
Здесь приложение клиента сначала объявляет делегат, соответствующий сигнатуре метода GetAllAutos() удаленного типа CarProvider. После создания делегата имя вызываемого метода (GetAllAutos) передается ему, как обычно. Потом запускается метод BeginInvoke(), сохраняется результирующий интерфейс IAsyncResult и имитируется какая-то работа на стороне клиента (напомним, что свойство IAsyncResult.IsCompleted позволяет выяснить, завершил ли работу соответствующий метод). После завершения работы клиента вы получаете список List‹›, возвращенный методом CarProvider.GetAllAutos() в результате вызова члена EndInvoke(), и передаете каждый объект JamesBondCar статической вспомогательной функции с именем UseCar().
public static void UseCar(JamesBondCar j) {
Console.WriteLine("Может ли машина летать"? {0}", j.isFlightWorthy);
Console.WriteLine("Может ли машина плавать? {0}", j.isSeaWorthy);
}
Снова подчеркнем, что красота использования типа делегата .NET заключается в том, что форма асинхронного вызова удаленных методов оказывается аналогичной форме вызова локальных методов.
Исходный код. Проект AsynсWKOCarProviderClient размещен в подкаталоге, соответствующем главе 18.