UNIX: разработка сетевых приложений - Страница 5
Fri Jan 12 14:27:52 1996rnгде
rnreadreadВ приведенном примере конец записи обозначается сервером, закрывающим соединение. Эта технология используется также версией 1.0 протокола передачи гипертекста (Hypertext Transfer Protocol, HTTP). Существуют и другие способы обозначения конца записи. Например, протокол передачи файлов (File Transfer Protocol, FTP) и простой протокол передачи почты (Simple Mail Transfer Protocol, SMTP) обозначают конец записи 2-байтовой последовательностью, состоящей из символов ASCII возврата каретки и перевода строки. Служба вызова удаленных процедур (Remote Procedure Call, RPC) и система именования доменов (Domain Name System, DNS) помещают перед каждой записью, отсылаемой по протоколу TCP, двоичное число, соответствующее длине этой записи. Здесь важно осознать, что протокол TCP сам по себе не предоставляет никаких меток записей: если приложение хочет отделять записи одну от другой, оно должно делать это самостоятельно, и для этого имеются стандартные методы.
26exitКак уже говорилось, пока мы лишь выделили наиболее важные моменты, детальным исследованием которых займемся в дальнейшем.
1.3. Независимость от протокола
Наша программа, представленная в листинге 1.1, является зависимой от протокола (protocol dependent) IPv4. Мы выделяем и инициализируем структуру
sockaddr_insocketAF_INETЕсли мы хотим изменить программу так, чтобы она работала по протоколу IPv6, мы должны изменить код. В листинге 1.2 показана новая версия программы с соответствующими изменениями, отмеченными полужирным шрифтом.
Листинг 1.2. Версия листинга 1.1 для IPv6
//intro/daytimetcpcliv6.с 1 #include "unp.h" 2 int 3 main(int argc, char **argv) 4 { 5 int sockfd, n; 6 char recvline[MAXLINE + 1]; 7 struct sockaddr_in6 servaddr; 8 if (argc != 2) 9 err_quit("usage: a.out <Ipaddress>");10 if ((sockfd = socket(AF_INET6, SOCK_STREAM, 0)) < 0)11 err_sys("socket error");12 bzero(&servaddr, sizeof(servaddr));13 servaddr.sin6_family = AF_INET6;14 servaddr.sin6_port = htons(13); /* сервер времени и даты */15 if (inet_pton(AF_INET6, argv[1], &servaddr.sin6_addr) <= 0)16 err_quit("inet_pton error for %s", argv[1]);17 if (connect(sockfd, (SA*)&servaddr, sizeof(servaddr)) < 0)18 err_sys("connect error");19 while ((n = read(sockfd, recvline, MAXLINE)) > 0) {20 recvline[n] = 0; /* символ конца строки */21 if (fputs(recvline, stdout) == EOF)22 err_sys("fputs error");23 }24 if (n < 0)25 err_sys("read error");26 exit(0);27 }Изменились только пять строк, но в результате мы все равно получили программу, зависимую от протокола, в данном случае — от протокола IPv6. Лучше сделать программу независимой от протокола (protocol independent). В листинге 11.3 представлена независимая от протокола версия этого клиента, основанная на вызове
getaddrinfotcp_connectДругим недостатком наших программ является то, что пользователь должен вводить IP-адрес сервера в точечно-десятичной записи (например, 206.168.112.219 для версии IPv4). Людям проще работать с именами, чем с числами (например,
www.unpbook.com1.4. Обработка ошибок: функции-обертки
В любой реальной программе существенным моментом является проверка каждого вызова функции на предмет возвращаемой ошибки. В листинге 1.1 мы проводим поиск ошибок в вызовах функций
socketinet_ptonconnectreadfputserr_quiterr_sys