Создание объекта с возможностью удаленного доступа

В данной лабораторной работе необходимо создать объект, который будет размещаться на сервере и вызываться клиентом через канал HTTP с использованием форматера канала HTTP по умолчанию (форматера SOAP). Так как на обоих концах взаимодействия имеется .NET, также легко можно бы использовать канал TCP и двоичный форматер.

Целью данной лабораторной работы является демонстрация вызова функциональности сервера через вызов удаленного доступа.

В данной работе мы создадим объект с простым вычислительным методом и покажем его исполнение на сервере. Необходимо разработать следующие компоненты приложения:

- вызываемый удаленно объект;

- сервер, который будет содержать этот объект удаленного доступа;

- клиент, который будет использовать объект удаленного доступа.

Архитектура приложения представлена на рисунке.

Клиент будет создавать объект Order и будет думать, что он локален по отношению к процессу клиента. Однако при использовании объекта Order структура удаленного доступа будет упаковывать вызов в сообщение SOAP и передавать его через HTTP -порт 8080 на сервер. Сервер будет распаковывать сообщение SOAP, исполнять метод и запаковывать сообщение для возврата сообщения SOAP клиенту. Клиент будет распаковывать ответный вызов, и получать результаты вызова метода.

Перед тем, как мы реализуем сервер и клиента удаленного доступа, нам требуется создать объект или набор объектов, которые можно будет вызывать через удалённый доступ. Чтобы предоставить наш класс, нам требуется определить, реализовывать ли наш объект как MarshalByRefObject или как сериализуемый. Так как мы хотим использовать логику исполнения нашего объекта на сервере, мы должны создать объект MarshalByRefObject.

Следующий код предоставляет базовый объект, который будет доступен через удаленный доступ:

using System;

using System.Collections.Generic;

using System.Text;

 

namespace Order

{

public class Order:MarshalByRefObject

{

private DateTime m_creation;

 

public Order()

{

m_creation = DateTime.Now;

}

 

public String GetMachineName()

{

Console.WriteLine("Order.GetMachineName() called.");

return String.Format("Order object created at: {0} on {1}",

m_creation.ToLongTimeString(), Environment.MachineName);

}

 

public Double CalculateItem(Double cost, int TaxCode)

{

Console.WriteLine("Order.CalculateItem() called.");

switch (TaxCode)

{

case 0:

return cost;

case 1:

return cost*1.06;

case 2:

return cost*1.12;

default:

throw new Exception("Error!");

}

}

}

}

 

Заметьте, что логика объекта не изменяется из-за того, что мы планируем сделать его доступным через удаленный доступ. Единственным отличием между этим и стандартным объектами заключается в том, что он наследуется от MarshalByRefObject, что делает его доступным через уделенный доступ.

Мы храним время создания объекта, чтобы можно было определить, когда объект был создан. Свойство GetMachineName возвращает имя машины, на которой выполняется код объекта. Имя машины докажет, что объект на самом деле исполняется в процессе сервера. Метод CalculateItem() является небольшим примером бизнес-логики, которая может находиться в классе Order. Очевидно, что для целей демонстрации мы создаем простую функцию. Если бы это был настоящий объект Order, то метод был бы гораздо сложнее; возможно, метод CalculateItem() возвращал бы документ XML, определяющий весь заказ на покупку.

Хорошим правилом при создании объектов удаленного доступа является отсутствие у них состояний. Так как объекты удаленного доступа будут вызываться из многих клиентских приложений, мы не хотим хранить открытые экземпляры для каждого пользователя.