-->

UNIX: разработка сетевых приложений

На нашем литературном портале можно бесплатно читать книгу UNIX: разработка сетевых приложений, Стивенс Уильям Ричард-- . Жанр: ОС и Сети. Онлайн библиотека дает возможность прочитать весь текст и даже без регистрации и СМС подтверждения на нашем литературном портале bazaknig.info.
UNIX: разработка сетевых приложений
Название: UNIX: разработка сетевых приложений
Дата добавления: 16 январь 2020
Количество просмотров: 373
Читать онлайн

UNIX: разработка сетевых приложений читать книгу онлайн

UNIX: разработка сетевых приложений - читать бесплатно онлайн , автор Стивенс Уильям Ричард

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

Внимание! Книга может содержать контент только для совершеннолетних. Для несовершеннолетних чтение данного контента СТРОГО ЗАПРЕЩЕНО! Если в книге присутствует наличие пропаганды ЛГБТ и другого, запрещенного контента - просьба написать на почту [email protected] для удаления материала

Перейти на страницу:

39   }

40  }

41  exit(0);

42 }

Эта программа корректно работает на узле с единственным IP-адресом. Если запустить программу из листинга 11.1 на узле с четырьмя IP-адресами, то получим:

freebsd % <b>hostent cnn.com</b>

official hostname: cnn.com

address: 64.236.16.20

address: 64.236.16.52

address: 64.236.16.84

address: 64.236.16.116

address: 64 236.24.4

address: 64.236.24.12

address: 64.236.24.20

address: 64.236.24.28

Но если запустить программу из листинга Д.4 на том же узле, в выводе будет только первый IP-адрес:

freebsd % <b>hostent2 cnn.com</b>

official hostname: cnn.com

address: 64.236.24.4

name = www1.cnn.com

Проблема заключается в том, что две функции,

gethostbyname
и
gethostbyaddr
, совместно используют одну и ту же структуру
hostent
, как было показано в разделе 11.18. Когда наша новая программа вызывает функцию
gethostbyaddr
, она повторно использует данную структуру вместе с областью памяти, на которую структура указывает (массив указателей
h_addr_list
), стирая три оставшиеся IP-адреса, возвращаемые функцией
gethostbyname
.

11.2. Если ваша система не поддерживает повторно входимую версию функции

gethostbyaddr
(см. раздел 11.19), то прежде чем вызывать функцию
gethostbyaddr
, вам следует создать копию массива указателей, возвращаемых функцией
gethostbyname
, и данных, на которые указывает этот массив.

11.3. Сервер

chargen
отправляет клиенту данные до тех пор, пока клиент не закрывает соединение (то есть пока вы не завершите выполнение клиента).

11.4. Эта возможность поддерживается некоторыми распознавателями, но переносимая программа не может использовать ее, потому что POSIX никак ее не оговаривает. В листинге Д.5 приведена измененная версия. Порядок тестирования строки с именем узла имеет значение. Сначала мы вызываем функцию

inet_pton
, поскольку она обеспечивает быстрый тест «внутри памяти» (in-memory) для проверки, является ли строка допустимым IP-адресом в точечно-десятичной записи. Только если тест заканчивается неудачно, мы запускаем функцию
gethostbyname
, которая обычно требует некоторых сетевых ресурсов и времени.

Если строка является допустимым IP-адресом в точечно-десятичной записи, мы создаем свой массив указателей (

addrs
) на один IP-адрес, оставив без изменений цикл, использующий
pptr
.

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

memcpy
в листинге 11.2 на вызов функции
memmove
, так как при вводе IP-адреса в точечно-десятичной записи исходное и конечное поля в данном вызове одинаковые.

Листинг Д.5. Допускаем как использование IP-адреса в точечно-десятичной записи, так и задание имени узла, номера порта или имени службы

//names/daytimetcpcli2.c

 1 #include &quot;unp.h&quot;

 2 int

 3 main(int argc, char **argv)

 4 {

 5  int sockfd, n;

 6  char recvline[MAXLINE + 1];

 7  struct sockaddr_in servaddr;

 8  struct in_addr **pptr, *addrs[2];

 9  struct hostent *hp;

10  struct servent *sp;

11  if (argc != 3)

12   err_quit(&quot;usage: daytimetcpcli2 &lt;hostname&gt; &lt;service&gt;&quot;);

13  bzero(&amp;servaddr, sizeof(servaddr));

14  servaddr.sin_family = AF_INET;

15  if (inet_pton(AF_INET, argv[1], &amp;servaddr.sin_addr) == 1) {

16   addrs[0] = &amp;servaddr.sin_addr;

17   addrs[1] = NULL;

18   pptr = &amp;addrs[0];

19  } else if ((hp = gethostbyname(argv[1])) != NULL) {

20   pptr = (struct in_addr**)hp-&gt;h_addr_list;

21  } else

22   err_quit(&quot;hostname error for %s: %s&quot;, argv[1], hstrerror(h_errno));

23  if ((n = atoi(argv[2])) &gt; 0)

24   servaddr.sin_port = htons(n);

25  else if ((sp = getservbyname(argv[2], &quot;tcp&quot;)) != NULL)

26   servaddr.sin_port = sp-&gt;s_port;

27  else

28   err_quit(&quot;getservbyname error for %s&quot;, argv[2]);

29  for (; *pptr != NULL; pptr++) {

30   sockfd = Socket(AF_INET, SOCK_STREAM, 0);

31   memmove(&amp;servaddr.sin_addr, *pptr, sizeof(struct in_addr));

32   printf(&quot;trying %sn&quot;,

33    Sock_ntop((SA*)&amp;servaddr, sizeof(servaddr)));

34   if (connect(sockfd, (SA*)&amp;servaddr, sizeof(servaddr)) == 0)

35    break; /* успех */

36   err_ret(&quot;connect error&quot;);

37   close(sockfd);

38  }

39  if (*pptr == NULL)

40   err_quit(&quot;unable to connect&quot;);

Перейти на страницу:
Комментариев (0)
название