Симон Робинсон - C# для профессионалов. Том II
- Название:C# для профессионалов. Том II
- Автор:
- Жанр:
- Издательство:Лори
- Год:2003
- Город:Москва
- ISBN:5-85582-187-0
- Рейтинг:
- Избранное:Добавить в избранное
-
Отзывы:
-
Ваша оценка:
Симон Робинсон - C# для профессионалов. Том II краткое содержание
Платформа .NET предлагает новую среду, в которой можно разрабатывать практически любое приложение, действующее под управлением Windows, а язык C# — новый язык программирования, созданный специально для работы с .NET.
В этой книге представлены все основные концепции языка C# и платформы .NET. Полностью описывается синтаксис C#, приводятся примеры построения различных типов приложений с использованием C# — создание приложений и служб Windows, приложений и служб WWW при помощи ASP.NET, а также элементов управления Windows и WWW Рассматриваются общие библиотеки классов .NET, в частности, доступ к данным с помощью ADO.NET и доступ к службе Active Directory с применением классов DirectoryServices.
Для кого предназначена эта книгаЭта книга предназначена для опытных разработчиков, возможно, имеющих опыт программирования на VB, C++ или Java, но не использовавших ранее в своей работе язык C# и платформу .NET. Программистам, применяющим современные технологии, книга даст полное представление о том, как писать программы на C# для платформы .NET.
Основные темы книги• Все особенности языка C#
• C# и объектно-ориентированное программирование
• Приложения и службы Windows
• Создание web-страниц и web-служб с помощью ASP NET
• Сборки .NET
• Доступ к данным при помощи ADO NET
• Создание распределённых приложений с помощью NET Remoting
• Интеграция с COM, COM+ и службой Active Directory
C# для профессионалов. Том II - читать онлайн бесплатно полную версию (весь текст целиком)
Интервал:
Закладка:
Класс Client
представляет клиентское приложение. Этот класс создает экземпляр класса EventSink
и регистрирует метод StatusHandler()
этого класса как обработчика для делегата в удаленном объекте. EventSink
должен быть удаленным объектом, подобным классу RemoteClass
, так как этот класс также будет вызываться через сеть.

Удаленный объект
Класс удаленного объекта реализуется в файле RemoteObject.cs
. Класс удаленного объекта должен выводиться из MarshalByRefObject
так, как было показано в предыдущих примерах. Чтобы сделать возможным для клиента регистрацию обработчика событий, который вызывается из удаленного объекта, необходимо объявить внешнюю функцию с помощью ключевого слова delegate
. Мы объявляем делегата StatusEvent()
с двумя аргументами: sender
(поэтому клиент знает об объекте, который порождает событие) и переменную типа StatusEventArgs
. В класс аргумента помещаем всю дополнительную информацию, которую необходимо послать клиенту.
Метод, который реализуется на стороне клиента, требует строгих ограничений. Он может иметь только входные параметры, возвращаемые типы, при этом параметры ref
и out
недопустимы; а типы аргументов должны быть либо [Serializable]
, либо удаленными (выводимыми из MarshalByRefObject
):
public delegate void StatusEvent(object sender, StatusEventArgs e);
public class RemoteObject : MarshalByRefObject {
Внутри класса RemoteObject
объявляется экземпляр функции делегата Status
, модифицированный ключевым словом event
. Клиент должен добавить обработчик событий в событие Status
, чтобы получить статусную информацию из удаленного объекта:
public class RemoteObject : MarshalByRefObject {
public RemoteObject() {
Console.WriteLine("RemoteObject constructor called");
}
public event StatusEvent Status;
В методе LongWorking()
проверяется, что обработчик событий регистрируется прежде, чем событие порождается с помощью Status(this, е)
. Чтобы удостовериться, что событие порождается асинхронно, мы получаем событие в начале метода перед выполнением Thread.Sleep()
и после Sleep
:
public void LongWorking(int ms) {
Console.WriteLine("RemoteObject: LongWorking() Started");
StatusEventArgs e = new StatusEventArgs("Message for Client: LongWorking() Started");
// породить событие
if (Status != null) {
Console.WriteLine("RemoteObject: Firing Starting Event");
Status(this, e);
}
System.Threading.Thread.Sleep(ms);
e.Message = "Message for Client: LongWorking() Ending"; // породить событие окончания
if (Status != null) {
Console.WriteLine("RemoteObject: Firing Ending Event");
Status(this, e);
}
Console.WriteLine("RemoteObject: LongWorking() Ending");
}
}
Аргументы событий
Мы видели в классе RemoteObject
, что класс StatusEventArgs
используется как аргумент для делегата. С помощью атрибута [Serializable]
экземпляр этого класса может передаваться от сервера клиенту. Мы используем простое свойство типа string для пересылки клиенту сообщения:
[Serializable]
public class StatusEventArgs {
public StatusEventArgs(string m) {
message = m;
}
public string Message {
get {
return message;
}
set {
message = value;
}
}
private string message;
}
Сервер
Сервер реализуется внутри консольного приложения. Мы ожидаем только, чтобы пользователь завершил работу сервера после чтения конфигурационного файла и настройки канала и удаленного объекта:
using System;
using System.Runtime.Remoting;
namespace Wrox.ProfessionalCSharp {
class Server {
static void Main(string[] args) {
RemotingConfiguration.Configure("Server.exe.config");
Console.WriteLine("Hit to exit");
Console.ReadLine();
}
}
}
Способ создания конфигурационного файла сервера Server.exe.config
мы уже обсуждали. Существует только один важный момент. Так как клиент сначала регистрирует обработчик событий и после этого вызывает удаленный метод, то удаленный объект должен сохранять состояние клиента. Можно использовать с событиями объекты SingleCall
, поэтому класс RemoteObject
конфигурируется в виде активированного клиентом типа:
Приемник событий
Приемник событий реализует обработчик StatusHandler()
, который определен в делегате. Как ранее отмечалось, метод может иметь только входные параметры и возвращать только void
. Это в точности соответствует требованиям методов [OneWay]
, как мы видели ранее при рассмотрении асинхронной удаленной работы. StatusHandler()
будет вызываться асинхронно. Класс EventSink
должен также наследовать из класса MarshalByRefObject
, чтобы сделать его удаленным, так как он будет вызывать с сервера удаленным образом:
using System;
using System.Runtime.Remoting.Messaging;
namespace Wrox.ProfessionalCSharp; {
public class EventSink MarshalByRefObject {
public EventSink() { }
[OneWay]
public void StatusHandler(object sender, StatusEventArgs e) {
Сonsole.WriteLine("EventSink: Event occurred: " + e.Message);
}
}
}
Клиент
Клиент читает конфигурационный файл клиента с помощью класса RemotingConfiguration
. Так было со всеми клиентами, которые создавались до сих пор. Клиент создает локально экземпляр удаленного класса приемника EventSink
. Метод, который должен вызываться из удаленного объекта на сервере, передается в удаленный объект:
using System;
using System.Runtime.Remoting;
namespace Wrox.ProfessionalCSharp {
class Client {
static void Main(string[] args) {
RemotingConfiguration.Configure("Client.exe.config");
Различие начинается здесь. Мы должны создать локально экземпляр удаленного класса приемника EventSink
. Так как этот класс не будет конфигурироваться элементом , то его экземпляр создается локально. Затем мы получаем экземпляр класса удаленного объекта RemoteObject
. Этот класс конфигурируется в элементе , поэтому его экземпляр создается на удаленном сервере:
EventSink sink = new EventSink();
RemoteObject obj = new RemoteObject();
Теперь можно зарегистрировать метод обработчика объекта EventSink
на удаленном объекте. StatusEvent
является именем делегата, который был определен на сервере. Метод StatusHandler()
имеет те же самые аргументы, которые определены в StatusEvent
.
Вызывая метод LongWorking()
, сервер будет делать обратный вызов в методе StatusHandler()
в начале и в конце метода:
// зарегистрировать клиентский приемник на сервере — подписаться
// на событие
obj.Status += new StatusEvent(sink.StatusHandler);
obj.LongWorking(5000);
Теперь мы более не заинтересованы в получении событий с сервера и отменяем подписку на событие. Следующий раз при вызове LongWorking()
никакие события не будут получены.
Интервал:
Закладка: