Опубликован: 10.10.2010 | Уровень: специалист | Доступ: свободно
Лекция 3:

Использование API java.net

< Лекция 2 || Лекция 3: 123 || Лекция 4 >
Аннотация: В лекции рассматривается API, реализованное в пакете java.net и предназначенное для создания приложений, которые используют протоколы TCP и UDP

Рабочий каталог расположен в Practice.

Итак, мы собираемся написать наше первое распределенное приложение. Способ, с помощью которого мы собираемся это сделать, - самый простой. Раз нам нужно передавать данные между компонентами нашего приложения по сети, следовательно, этому нужно научиться. Первый способ, который мы используем для решения задачи, будет основываться на входящем в состав пакета java.net API. Этот пакет предоставляет возможность для реализации сетевого взаимодействия с применением одного из двух транспортных протоколов: UDP и TCP.

Архитектура протоколов TCP/IP, известная как набор протоколов TCP/IP, возникла в результате исследований в области протоколов и разработок, выполнявшейся в экспериментальной сети с коммутацией пакетов под названием ARPANET, которая была основана Управлением перспективных исследовательских программ Министерства обороны США ( DARPA )[3.26]. Этот набор протоколов состоит из большого собрания протоколов, изданных Координационным советом по сети Internet (IAB) в качестве стандартов для Internet.

В общем можно сказать, что в обмене информацией всегда принимают участие три агента: приложения, узлы (компьютеры) и сети. К приложениям относятся программы, предназначенные, например, для передачи файлов или электронной почты. Приложения, о которых идет речь, являются распределенными приложениями, включающими в себя обмен данными между двумя и более компьютерными системами. Компьютеры соединены между собой в сети, и предназначенные для обмена данные передаются по сети от одного компьютера к другому. Таким образом, передача данных от одного приложения другому (или от одного компонента другому в рамках одного распределенного приложения) включает в себя, во-первых, получение данных тем узлом, на котором находится приложение-адресат, и, во-вторых, получение данных именно тем выполняющимся на узле-адресате приложением, которому они предназначены (одновременно на узле могут выполняться несколько приложений).

Ввиду этого в задаче обмена информацией выделяют пять основных относительно независимых уровней:

  • физический уровень;
  • уровень доступа к сети (сетевой);
  • межсетевой уровень;
  • транспортный уровень;
  • уровень приложений.

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

Уровень доступа к сети связан с обменом данными между конечной системой (рабочей станцией, сервером и т.д.) и сетью, к которой подсоединена эта система. Узел-отправитель должен передать в сеть адрес узла-адресата, чтобы сеть могла направить данные по месту требования.

Уровень доступа к сети рассматривается в связи с доступом к сети и маршрутизацией данных между двумя подключенными к сети оконечными системами. В тех случаях, когда устройства подключены к разным сетям, нужны дополнительные процедуры, позволяющие данным переходить из одной сети в другую. Такие функции относятся к межсетевому уровню. На этом уровне функции межсетевой маршрутизации предоставляются с помощью Internet -протокола ( IP ). Internet -протокол может быть реализован не только в конечных системах, но и в маршрутизаторах.

Независимо от природы приложений обмен данными должен быть надежным, т.е. хотелось бы иметь уверенность в том, что все данные попали к приложению-адресату и что эти данные получены в том порядке, в котором они были отправлены. Механизмы обеспечения надежности, по сути, независимы от природы приложений, таким образом, имеет смысл выделить такие механизмы в один общий уровень, совместно используемый всеми приложениями. Этот уровень называется транспортным. Чаще всего для него применяется протокол управления передачей ( Transmission Control Protocol - TCP ).

Наконец, в уровень приложений заложена логика, необходимая для обеспечения работы различных пользовательских приложений. Приложению каждого вида нужен отдельный модуль, специально предназначенный для этого приложения. Именно разработкой уровня приложения мы в дальнейшем и займемся.

Итак, чтобы обмен информацией был возможен, каждый элемент системы должен иметь уникальный адрес. Фактически, нужно указать два уровня адресации. Каждый узел сети должен обладать своим уникальным глобальным сетевым адресом, это позволит доставить данные соответствующему узлу. Каждый процесс (компонент) на узле должен иметь адрес, который был бы уникальным в пределах этого узла, что даст возможность транспортному протоколу ( TCP ) доставить данные нужному процессу. Этот адрес - адрес процесса - в терминологии протоколов семейства TCP/IP называют портом.

Для большинства приложений, выполняющихся в рамках архитектуры протокола TCP/IP, протоколом транспортного уровня выступает TCP. Этот протокол обеспечивает надежное соединение для передачи данных от одного приложения к другому. Кроме него существует еще один широко используемый протокол транспортного уровня, входящий в набор протоколов TCP/IP: пользовательский протокол датаграмм ( User Datagram Protocol - UDP ). Протокол UDP предоставляет сервис без установления соединения, предназначенный для процедур на уровне приложения; он не гарантирует доставку, сохранение последовательности при приеме переданных пакетов или защиту от их дублирования. Он позволяет процессу отправлять сообщения другим процессам, с помощью минимального протокольного механизма. Протокол UDP выполняет крайне ограниченный набор функций, так как работает без установления соединения. По сути, он добавляет к протоколу IP некоторые возможности адресации портов.

Рассмотрим вкратце возможности, которые предоставляет нам java и ее пакет java.net.

UDP

Начнем мы с API, использующего протокол UDP. UDP является протоколом, применяющим для передачи датаграммы. Как уже говорилось, этот протокол не является надежным, поскольку сообщения, передаваемые с его помощью, могут теряться или приходить в другой последовательности, отличной от последовательности их отправки. Таким образом, для обеспечения надежной передачи необходимо организовывать надстройку над этим протоколом, обеспечивающую, например, нумерацию пакетов, повторную передачу пакетов при истечении времени ожидания и т.д. Длина одного сообщения (одной датаграммы) при использовании этого протокола ограничена 65536 байтами (причем многие реализации вообще ограничивают размер датаграммы 8К), в случае необходимости пересылки порции данных большего размера они должны быть разбиты на куски отправителем и снова собраны получателем. Передача сообщения - не блокирующая, прием - блокирующий с возможностью прерывания по истечении времени ожидания.

Для работы с UDP в пакете java.net определены следующие классы.

  • DatagramPacket (датаграмма). Конструктор этого класса принимает массив байт и адрес процесса-получателя ( IP -адрес узла и порт). Класс предназначен для представления единичной дата-граммы (сообщения). Этот класс используется как для создания сообщения с целью последующей его отправки, так и при приеме сообщения (функция приема возвращает экземпляр этого класса).
  • DatagramSocket.Предназначен для посылки/приема UDP -дата-грамм. Один из конструкторов принимает в качестве аргумента порт, с которым связывается сокет, другой конструктор, без аргументов, задействует в качестве порта первый попавшийся свободный порт. Класс имеет методы send и receive, для, соответственно, передачи и приема датаграмм. Метод setSoTimeout устанавливает тайм-аут для операций сокета.

Ниже приведены две простые программы, использующие рассмотренные механизмы для организации взаимодействия.

UDPClient

Первая программа создает сокет, соединяется с сервером (порт 6789), пересылает ему сообщение и ждет ответа (пример 3.1).

1  import java.net.*;
2  import java.io.*;
3  public class UDPClient{
4  public static void main(String args[]){
5    // аргументы - сообщение и адрес сервера
6    try {
7       DatagramSocket aSocket = new DatagramSocket(); // create socket
8       byte [] message = args[0].getBytes();
9       InetAddress aHost = InetAddress.getByName(args[1]); // DNS lookup
10      int serverPort = 6789;
11      DatagramPacket request =
12      new DatagramPacket(message,   args[0].length(), aHost, serverPort);
13      aSocket.send(request);
14      //send message
15      byte[] buffer = new byte[1000];
16      DatagramPacket reply = new DatagramPacket(buffer, buffer.length);
17      aSocket.receive(reply);
18      //wait for reply
19      System.out.println("Reply: " + new String(reply.getData()));
20      aSocket.close();
21   } catch (SocketException e){ System.out.println("Socket: " + e.getMessage());
22  // socket creation failed
23  } catch (IOException e){ System.out.println("IO: " + e.getMessage());
24  // ошибка при приеме
25  }
26  }
27  }
Листинг 3.1. Клиент

Поскольку это первые наши программы, прокомментируем их более подробно.

В первых строках (строки 1,2) происходит импорт необходимых библиотек (пакетов) классов - в нашем случае это java.net и java.io. Первый пакет содержит необходимые нам классы для работы с UDP, второй определяет необходимые классы ввода/вывода.

Наш класс (строка 3) называется UDPClient,он содержит единственный статический метод main (строка 4), являющийся точкой входа в программу. Метод принимает массив аргументов командной строки, переданных программе при запуске. В качестве аргументов для нашей программы должны быть переданы: строка сообщения (которое будет оправлено на сервер) и адрес узла, на котором запущена программа, - сервер (порт не передается, поскольку в нашем случае он заранее известен). Далее создается сокет для передачи сообщения (строка 7), затем происходит разрешение переданного в качестве аргумента командной строки имени узла сервера в адрес (строка 9). Затем создается датаграмма (строки 11-12), в конструкторе которой передается массив, составляющий передаваемые данные, адрес узла, на котором выполняется сервер и порт (который нам известен заранее). Пакет передается на сервер (строка 13). Обратите внимание - адрес получателя определяется в пакете, а не в сокете, через который мы его передаем. Т.е. один и тот же сокет может быть использован для передачи пакетов разным узлам, и этот же сокет может применяться для приема пакета от сервера (строка 17). После использования сокет должен быть закрыт (строка 20).

UDPServer

Другая программа (сервер) создает сокет и обслуживает запросы клиента (пример 3.2).

1  import java.net.*;
2  import java.io.*;
3  public class UDPServer{
4    public static void main(String args[]) {
5      DatagramSocket aSocket = null;
6      try{
7         aSocket = new DatagramSocket(6789); // create socket at port
8         byte[] buffer = new byte[1000];
9         while(true){
10           DatagramPacket request = new DatagramPacket(buffer, buffer.length);
11           aSocket.receive(request);
12           DatagramPacket reply =
13           new DatagramPacket(request.getData(),  request.getLength(),
14           request.getAddress(), request.getPort());
15           aSocket.send(reply);
16       }
17     } catch (SocketException e) {System.out.println("Socket: " + e.getMessage());
18    // socket creation failed
19 } catch (IOException e) {System.out.println("IO: " + e.getMessage());
20 } finally {if(aSocket != null) aSocket.close();}
21 }
22 }
Листинг 3.2. Сервер

Компиляция

В том случае, когда при установке пакета jdk в системе были установлены соответствующие пути, для компиляции примера нужно выполнить следующие действия:

  • перейти в каталог, где сохранены файлы UDPClient.java и UDPServer.java ;
  • набрать в командной строке команду компиляции javac *.java.

После компиляции в текущем каталоге должны появиться два файла: UDPClient.class и UDPServer.class, которые представляют собой наши классы, откомпилированные в байт-код.

Запуск приложения

Первым запускается сервер, для его запуска необходимо, находясь в каталоге, где лежат исполняемые файлы, выполнить команду:

java UDPServer
Вывод сервера

Рис. 3.1. Вывод сервера
Вывод клиента

Рис. 3.2. Вывод клиента

После запуска сервер переходит в режим ожидания сообщений от клиента. Поскольку нами не предусмотрено никакого механизма останова сервера, выгрузить его может либо произошедшая ошибка, либо насильственное прерывание процесса.

Клиент запускается следующим образом:

java UDPClient Hello! 127.0.0.1

Первый параметр определяет строку, которая будет передана серверу, второй - адрес узла, на котором запущен сервер. В данном случае и клиент, и сервер запущены на одной машине, поэтому в качестве адреса используется адрес 127.0.0.1 ( localhost ).

После запуска клиент отправляет строку на сервер, печатает ее в консоли и завершает работу (рис. 3.2). Сервер же продолжает ожидать очередное подключение следующего клиента (рис. 3.1).

< Лекция 2 || Лекция 3: 123 || Лекция 4 >
Алмаз Мурзабеков
Алмаз Мурзабеков
Прохожу курс "Построение распределенных систем на Java" в третьей лекции где описывается TCPServer вылетает эта ошибка
"Connection cannot be resolved to a type"


Java version 1.7.0_05