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