Если Web-служба должна передавать приложению большие объемы данных, то лучше всего организовать это так, чтобы она возвращала указатель на файл с двоичными данными, а не поток данных в виде XML. В качестве показательного примера можно привести Web-службу, возвращающую фотографические изображения.
Хотя Web-служба и может возвратить изображение в виде массива байтов или целых чисел в кодировке XML, гораздо эффективнее возвратить строку с URL-адресом, указывающим на двоичный файл (например, //somewebserver/someshare/somedir/somefile.jpg), который может быть загружен мобильным приложением. Именно так действуют Web-браузеры; они загружают текст в удобочитаемой форме и компонуют информацию в виде HTML-документа, содержащего ссылки на двоичные файлы изображения, которые должны быть встроены в макет. Очень важно тщательно продумывать, в каком виде следует перемещать данные, и оптимизировать этот процесс, если объемы данных велики.
"Болтливость" в мобильных сетях обходится весьма недешево
Каждый запрос, посланный на сервер, означает необходимость начала, проведения и завершения диалога с сервером; следовательно, этот процесс сопровождается дополнительными накладными расходами по организации связи. Пять отдельных вызовов Web-служб, каждый из которых требует передачи одного параметра, гораздо менее эффективны, чем один запрос, содержащий пять параметров. Кроме того, учитывая прерывистый характер сеансов мобильной связи, использование пяти мелких запросов вместо одного более крупного повышает вероятность сбоя в процессе диалога между устройством и сервером. Это означает необходимость написания сложной логики, позволяющей выполнять необходимые восстановительные операции, если посередине такого диалога что-то пойдет не так. Применение одиночных вызовов Web-служб позволяет уменьшить вероятность сбоев и упростить логику восстановления связи в подобных случаях.
НА ЗАМЕТКУ
При использовании Web-служб справедлив тезис, в соответствии с которым лучше передавать двоичные данные в результате выполнения второго запроса, чем пытаться сразу же передавать большой поток XML-данных. Поскольку объем двоичных данных при преобразовании их к формату XML значительно возрастает, это приводит к увеличению длительности их передачи. При длительных временах передачи возрастает вероятность сбоев. Улучшенная модель предполагает выполнение одного вызова Web-службы для передачи всех данных, которые могут быть эффективно переданы в виде XML, и ряда последующих вызовов для передачи таких двоичных файлов, как файлы изображений. В листинге 15.11 представлен код, позволяющий загрузить файл с Web-сервера и сохранить его на устройстве. Если необходимо передать содержимое нескольких двоичных файлов, имеет смысл поэкспериментировать с объединением всех двоичных данных в один сжатый файл; такой объединенный файл может быть передан в виде двоичных данных и распакован на другом конце канала связи.
В листинге 15.9 приведен пример неэффективной организации работы с Web-службой с использованием многократных запросов и ответов. В листинге 15.10 представлен пример пакетной организации того же диалога, когда вся необходимая обработка осуществляется при помощи одного цикла "запрос/ответ". При любом удобном случае старайтесь уменьшать количество запросов, объединяя несколько мелких запросов в один более емкий.
Листинг 15.9. Неэффективная организация диалога с Web-службой, в которой используется множество вызовов
//--------------------------
//Создать и обработать заказ
//--------------------------
//0. Установить связь
int sessionID = someWebService.LogOn(userCredentials);
//1. Вызвать Web-службу и создать новый заказ
int orderID = someWebService.CreateNewOrder(sessionID, userInfo, productInfo);
//2. Вызвать Web-службу и передать информацию о платеже
someWebService.ConfirmPayment(sessionID, orderID, paymentInfo);
//3. Вызвать Web-службу и передать информацию о доставке
someWebService.ConfirmShipping(sessionID, orderID, shippingAddress);
//4. Вызвать Web-службу и завершить оформление заказа
someWebService.FinalizeOrder(sessionID, orderID);
Листинг 15.10. Группирование запросов в одном вызове Web-службы
//----------------------------------------------------------
//Создать и обработать заказ посредством группового запроса,
//включающего:
// 0. Начало связи
// 1. Создание нового заказа
// 2. Подтверждение платежа
// 3. Подтверждение доставки
// 4. Завершение оформления заказа
//-----------------------------------------------------------
//Сделать все за один раз
someWebService.BatchCreateOrder(userCredentials, userInfo, paymentInfo, shippingAddress);
В листинге 15.11 показан пример кода, который загружает с сервера двоичный файл и сохраняет его локально на устройстве. Этот код может пригодиться вам при загрузке с сервера файлов, подобных файлам изображений.
Листинг 15.11. Код для загрузки файла с Web-сервера
//-----------------------------------------------------------
//Осуществляет синхронную загрузку файла с Web-сервера
//и сохраняет его в локальной файловой системе
// [in] httpWhereFrom: URL-адрес файла
// (например, "http://someserver/somefile.jpg")
// [in] filenameWhereTo: Место, куда необходимо записать файл
// (например, "\\localfile.jpg")
//-----------------------------------------------------------
public void downloadFileToLocalStore(string httpWhereFrom, string filenameWhereTo) {
System.IO.FileStream myFileStream = null;
System.IO.Stream myHTTPResponseStream = null;
System.Net.WebRequest myWebRequest = null;
System.Net.WebResponse myWebResponse = null;
//Если файл, который мы хотим записать, уже существует, удалить его
if (System.IO.File.Exists(filenameWhereTo) == true) {
System.IO.File.Delete(filenameWhereTo);