WSAAsyncSelect
связывает с сообщением именно сокет, а не его дескриптор. Это означает, что если две программы используют один сокет (копия дескриптора которого была создана с помощью функции WSADuplicateSocket
), и первая программа вызывает WSAAsyncSelect
со своим дескриптором, а затем вторая — со своим, то вызов WSAAsyncSelect
, сделанный во второй программе, отменит вызов, сделанный в первой.
Для того, чтобы получать сообщения при готовности сокета как к чтению, так и к записи, нужно выполнить следующий код.
WSAAsyncSelect(S, Form1.Handle, WM_USER, FD_READ or FD_WRITE);
При необходимости с помощью or
можно комбинировать и большее число констант.
Из сказанного следует, что нельзя связать с разными событиями одного и того же сокета разные сообщения (или отправлять сообщения разным окнам), т. к. при одном вызове WSAAsyncSelect
можно передать только один дескриптор окна и один номер сообщения, а следующий вызов этой функции, с другим дескриптором и/или номером, отменит предыдущий. Функция WSAAsyncSelect
переводит сокет в неблокирующий режим. Если необходимо использовать асинхронный сокет в блокирующем режиме, после вызова WSAAsyncSelect
требуется перевести его в этот режим вручную.
Сообщение, которое связывается с асинхронным сокетом, может быть любым. Обычно его номер выбирают от WM_USER
и выше, чтобы исключить путаницу со стандартными сообщениями.
При получении сообщения его параметр wParam
содержит дескриптор сокета, на котором произошло событие. Младшее слово lParam
содержит произошедшее событие (одну из констант FD_XXX
), а старшее слово — код ошибки если она произошла. Для выделения кода события и кода ошибки из lParam
в библиотеке WinSock предусмотрены макросы WSAGETSELECTEVENT
и WSAGETSELECTERROR
соответственно. В модуле WinSock они заменены функциями WSAGetSelectEvent
и WSAGetSelectError
. Одно сообщение может информировать только об одном событии на сокете. Если произошло несколько событий, в очередь окна будет добавлено несколько сообщений.
Сокет, созданный при вызове функции accept
, наследует режим того сокета, который принял соединения. Таким образом, если сокет, находящийся в режиме ожидания подключения, является асинхронным, то и сокет, порожденный функцией accept
, будет асинхронным, и тот же набор его событий будет связан с тем же сообщением, что и у исходного сокета.
Рассмотрим подробнее каждое из перечисленных событий.
Событие FD_READ
возникает, когда во входной буфер сокета поступают данные (если на момент вызова WSAAsyncSelect
, разрешающего такие события, в буфере сокета уже есть данные, то событие также возникает). Как только соответствующее сообщение помещается в очередь окна, дальнейшая генерация таких сообщений для этого сокета блокируется, т. е. получение новых данных не будет приводить к появлению новых сообщений (при этом сообщения, связанные с другими событиями этого сокета или с событием FD_READ
других сокетов, будут по-прежнему помещаться при необходимости в очередь окна). Генерация сообщений снова разрешается после того, как будет вызвана функция для чтения данных из буфера сокета (это может быть функция recv
, recvfrom
, WSARecv
или WSARecvFrom
, мы в дальнейшем будем говорить только о функции recv
, потому что остальные ведут себя в этом отношении аналогично).
Если после вызова recv
в буфере асинхронного сокета остались данные, в очередь окна снова помещается это же сообщение. Благодаря этому программа может обрабатывать большие массивы по частям. Действительно, пусть в буфер сокета приходят данные, которые программа хочет забирать оттуда по частям. Приход этих данных вызывает событие FD_READ
, сообщение о котором помещается в очередь. Когда программа начинает обрабатывать это сообщение, она вызывает recv
и читает часть данных из буфера. Так как данные в буфере еще есть, снова генерируется сообщение о событии FD_READ
, которое ставится в конец очереди. Через некоторое время программа снова начинает обрабатывать это сообщение. Если и на этот раз данные будут прочитаны не полностью, в очередь снова будет добавлено такое же сообщение. И так будет продолжаться до тех пор, пока не будут прочитаны все полученные данные.