?7.4 無連接協議
和面向連接的協議比較起來,無連接協議的行為極為不同,因此,收發數據的方式也會有所不同。由于在和面向會話的服務器比較時,無連接接收端改動不大,所以我們先談談接收端(如果你愿意,也可稱之為服務器)。接下來再談發送端。
7.4.1 接收端
對于在一個無連接套接字上接收數據的進程來說,步驟并不復雜。先用s o c k e t或W S A S o c k e t建立套接字。再把這個套接字和準備接收數據的接口綁定在一起。這是通過b i n d函數(和面向會話的示例一樣)來完成的。和面向會話不同的是,我們不必調用l i s t e n和a c c e p t。相反,只需等待接收數據。由于它是無連接的,因此始發于網絡上任何一臺機器的數據報都可被接
收端的套接字接收。最簡單的接收函數是r e c v f r o m。它的定義如下:
int recvfrom(
???????SOCKET s,
???????char FAR * buf,
???????int len,
???????int flags,
???????struct sockaddr?FAR * from,
???????int FAR * fromlen
??????);
前面四個參數和r e c v是一樣的,其中包括標志M S G _ O O B和M S G _ P E E K。在使用無連接套接字時,和前面一樣,仍然提醒大家慎用M S G _ P E E K標志。對監聽套接字的具體協議來說,f r o m參數是一個S O C K A D D R結構,帶有指向地址結構的長度的f r o m l e n。這個A P I調用返回數
據時,S O C K A D D R結構內便填入發送數據的那個工作站的地址。
r e c v f r o m函數的Winsock 2版本是W S A R e c v F r o m。后者的原型是:
int WSARecvFrom(
?????????SOCKET s,
?????????LPWSABUF?lpBuffers,
?????????DWORD?dwBufferCount,
?????????LPDWORD?lpNumberOfBytesRecvd,
?????????LPDWORD?lpFlags,
?????????struct?sockaddr?FAR *?lpFrom,
?????????LPINT?lpFromlen,
?????????LPWSAOVERLAPPED?lpOverlapped,
?????????LPWSAOVERLAPPED_COMPLETTION_ROUTINE?lpComplettionROUTINE
????????);
兩者的差別在于接收數據的W S A B U F結構的用法上。你可以利用d w B u ff e r C o u n t為W S A R e c v F r o m提供一個或多個W S A B U F緩沖。提供多個緩沖,就可用發散集合了。讀取的字節總數返回在l p N u m b e r O f B y t e s R e c v d中。在調用W S A R e c v F r o m時,l p F l a g s參數可以是代表
無選項的0、M S G _ O O B、M S G _ P E E K或M S G _ PA RT I A L。這些標志還可以累加起來。如果在調用這個函數時,指定M S G _ PA RT I A L,提供者就知道返回數據,即使只收到了部分消息。調用返回之后,如果只收到部分消息,就會設置M S G _ PA RT I A L標志。再次返回之后,
W S A R e c v F r o m就會把l p F r o m參數(它是一個指向S O C K A D D R結構的指針)設為發送端的地址。再次提醒大家注意, l p F r o m L e n指向S O C K A D D R結構的長度,另外,在這個函數中,它還是一個指針,指向D W O R D。最后兩個參數, l p O v e r l a p p e d和l p C o m p l e t i o n R O U T I N E,用于
重疊I / O(我們將在下一章就此展開討論)。
在無連接套接字上接收(發送)數據的另一種方法是建立連接。聽起來有些奇怪吧,但事實的確如此。無連接的套接字一旦建立,便可利用S O C K A D D R參數(它被設為準備與之通信的遠程接收端地址)調用c o n n e c t或W S A C o n n e c t。但事實上并沒有建立連接。投入連接函數的套接字地址是與套接字關聯在一起的,如此一來,才能夠用R e c v和W S A R e c v來代替
r e c v f r o m和W S A R e c v F r o m。為什么呢?其原因是數據的始發處是已知的。如果在一次應用中,只和一個端點進行通信,便能很容易地與數據報套接字建立連接。
7.4.2 發送端
要在一個無連接的套接字上發送數據,有兩種選擇。第一種,也是最簡單的一種,便是建立一個套接字,然后調用s e n d t o或W S A S e n d To。我們先來講解s e n d t o函數,它的定義是這樣的:
?int sendto(?
????????SOCKET s,
????????const char FAR * buf,
????????int len,
????????int flags,
????????const struct sockaddr FAR * to,
????????int tolen
???????);
?除了b u f是發送數據的緩沖, l e n指明發送多少字節外,其余參數和r e c v f r o m的參數一樣。另外, t o參數是一個指向S O C K A D D R結構的指針,帶有接收數據的那個工作站的目標地址。
另外,也可以用Winsock 2函數W S A S e n d To。它的定義如下:
int WSASendTo(
????????SOCKET s,
????????LPWSABUF?lpBuffers,
????????DWORD?dwBufferCount,
????????LPWORD?lpNumberOfBytesSent,
????????DWORD?dwFlags,
????????const ?struct sockaddr FAR * lpTo,
????????int iToLen,
????????LPWSAOVERLAPPED?lpOverlapped,
????????LPWSAOVERLAPPED_COMPLETTION_ROUTINE?lpComplettionROUTINE
???????);
???????
再次提醒大家注意, W S A S e n d To和前一版本中的S e n d To函數類似。它把指向帶有發給接
收端的數據的指針當作l p B u ff e r s參數, d w B u ff e r C o u n t參數指明現在的結構是多少。我們可發送多個W S A B U F結構啟用發散集合I / O。在函數返回之前,W S A S e n d To把第四個參數l p N u m b e r
O f B y t e s S e n t設為真正發到接收端的字節數。l p To參數是針對具體協議的一個S O C K A D D R結構,并帶有接收端的地址。i To L e n參數是S O C K A D D R結構的長度。最后兩個參數, l p O v e r l a p p e d和l p C o m p l e t i o n R O U T I N E,用于重疊I / O(將在第8章討論)。
通過接收數據的方式,就可把一個無連接的套接字連接到一個端點地址,并可以用s e n d和W S A S e n d發送數據了。這種關聯一旦建立,就不能再用帶有地址的s e n d t o和W S A S e n d To,
除非這個地址是投到其中一個連接函數的地址,否則調用就會失敗,出現W S A E I S C O N N錯誤。
要取消套接字句柄與目標地址的關聯,唯一的辦法是在這個套接字句柄上調用c l o s e s o c k e t,
并建立一個新的套接字。
??????????????