Анимация
JavaScript
|
Главная Библионтека 17.10. Написание двусторонних клиентов 623 Комментарий При разветвлении (forking) процесса потомок получает копии всех открытых файловых манипуляторов родителя, включая сокеты. Вызывая close для файла или сокета, вы закрываете только копию манипулятора, принадлежащую текущему процессу. Если в другом процессе (родителе или потомке) манипулятор остался открытым, операционная система не будет считать файл или сокет закрытым. Рассмотрим в качестве примера сокет, в который посылаются данные. Если он открыт в двух процессах, то один из процессов может закрыть его, и операционная система все равно не будет считать сокет закрытым, поскольку он остается открытым в другом процессе. До тех пор пока он не будет закрыт другим процессом, процесс, читающий из сокета, не получит признак конца файла. Это может привести к недоразумениям и взаимным блокировкам. Чтобы избежать затруднений, либо вызовите close для незакрытых манипуляторов, либо воспользуйтесь функцией shutdown. Функция shutdown является более радикальной формой close - она сообщает операционной системе, что, даже несмотря на наличие копий манипулятора у других процессов, он должен быть помечен как закрытый, а другая сторона должна получить признак конца файла при чтении или SIGPIPE при записи. Числовой аргумент shutdown позволяет указать, какие стороны соединения закрываются. Значение О говорит, что чтение данных закончено, а другой конец сокета при попытке передачи данных должен получить SIGPIPE. Значение 1 говорит о том, что закончена запись данных, а другой конец сокета при попытке чтения данных должен получать признак конца файла. Значение 2 говорит о за-верщении как чтения, так и записи. Представьте себе сервер, который читает запрос своего клиента до конца файла и затем отправляет ответ. Если клиент вызовет close, сокет станет недоступным для ввода/вывода, поэтому ответ от сервера не доберется до клиента. Вместо этого клиент должен вызвать shutdown, чтобы закрыть соединение наполовину. print SERVER my request\n , # Отправить данные shutdown(SERVER, 1), # Отправить признак конца данных, # запись окончена Sanswer = <SERVER>, # Хотя чтение все еще возможно > Смотри также- Описание функций close и shutdown ьperlfunc{\); страница руководства shut-down{2) ващей системы (если есть). 17.10. Написание двусторонних клиентов Проблема Вы хотите написать полностью интерактивного клиента, в котором можно ввести строку, получить ответ, ввести другую строку, получить новый ответ и т. д. - словом, нечто похожее на telnet. Решение После того как соединение будет установлено, разветвите процесс. Один из близнецов только читает ввод и передает его серверу, а другой - читает выходные данные сервера и копирует их в поток вывода. Комментарий в отношениях «клиент/сервер» бывает трудно определить, чья сейчас очередь «говорить». Однозадачные решения, в которых используется версия select с четырьмя аргументами, трудны в написании и сопровождении. Однако нет причин игнорировать многозадачные решения, а функция fork кардинально упрощает эту проблему. После подключения к серверу, с которым вы будете обмениваться данными, вызовите fork. Каждый из двух идентичных (или почти идентичных) процессов выполняет простую задачу. Родитель копирует все данные, полученные из сокета, в стандартный вывод, а потомок одновременно копирует все данные из стандартного ввода в сокет. Исходный текст программы приведен в примере 17.4. Пример 17.4. biclient #1/usr/bm/perl -w # biclient - двусторонний клиент с разветвлением use strict, use 10 Socket, my (Shost, Sport, Skidpid, Shandle, Sline), unless (©ARGV == 2) { die usage SO host port } (Shost, Sport) = @ARGV, # Создать tcp-подключение для заданного хоста и порта Shandle = 10 Socket INET->new(Proto => tcp , PeerAddr => Shost PeerPort => Sport) or die can t connect to port Sport on Shost S , Shandle->autoflush(1), # Запретить буферизацию print STDERR [Connected to Shost Sport]\n , # Разделить программу на два идентичных процесса die can t fork S unless defined(Skidpid = fork()), If (Skidpid) { n Родитель копирует сокет в стандартный вывод while (defined (Sline = <Shandle>)) { print STDOUT Sline, kilK TERM => Skidpid), # Послать потомку SIGTERM 17.11. Разветвляющие серверы 625 else { # Потомок копирует стандартный ввод в сокет while (defined ($line = <STDIN>)) { print Shandle Sline, exit. Добиться того же эффекта с одним процессом намного труднее. Проще создать два процесса и поручить каждому простую задачу, нежели кодировать выполнение двух задач в одном процессе. Стоит воспользоваться преимуществами мультизадачности и разделить программу на несколько подзадач, как многие сложнейщие проблемы упрощаются на глазах. Функция kill в родительском блоке if нужна для того, чтобы послать сигнал потомку (в настоящее время работающему в блоке else), как только удаленный сервер закроет соединение со своего конца. Вызов kill в конце родительского блока ликвидирует порожденный процесс с заверщением работы сервера. Если удаленный сервер передает данные по байтам и вы хотите получать их немедленно, без ожидания перевода строки (которого вообще может не быть), замените цикл while родительского процесса следующей конструкцией: ту Sbyte, while (sysread(Shandle, Sbyte, 1) == 1) { print STDOUT Sbyte, t> Смотри также- Описание функций sysread и fork в perlfunc(l); документация по стандартному модулю IO::Socket; рецепты 16.5; 16.10; 17.11. 17.11. Разветвляющие серверы Проблема Требуется написать сервер, который для работы с очередным клиентом ответвляет специальный подпроцесс. Решение Ответвляйте подпроцессы в цикле accept и используйте обработчик $SIG{CHLD} для чистки потомков. # Создать сокет SERVER, вызвать bind и прослушивать use POSIX qw( sys wait h), sub REAPER { 1 until (-1 == waitpid(-1, WNOHANG)), $SIG{CHLD} = \&REAPER, # если $] >= 5 002 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 [ 202 ] 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 |