非同步 SOCKET 程式設計 - 傳送和接收資料 (轉)
原作者: DREW SIKORA
我本想把傳送和接收分開作為兩部分,但是最後我決定只略微解釋一下 FD_READ ,留下更多的時間來說明更復雜的 FD_WRITE , FD_READ 事件非常容易掌握. 當有資料傳送過來時, WinSock 會以 FD_READ 事件通知你, 對於每一個 FD_READ 事件, 你需要像下面這樣 recv() :
int bytes_recv = recv(wParam, &data, sizeof(data), 0);
基本上就是這樣, 別忘了修改上面的 wParam. 還有, 不一定每一次呼叫 recv() 都會接收到一個完整的資料包, 因為資料可能不會一次性全部傳送過來. 所以在開始處理接收到的資料之前, 最好對接收到的位元組數 ( 即 recv() 的返回值) 進行判斷, 看看是否收到的是一個完整的資料包.
FD_WRITE 相對來說就麻煩一些. 首先, 當你建立了一個連線時, 會產生一個 FD_WRITE 事件. 但是如果你認為在收到 FD_WRITE 時呼叫 send() 就萬事大吉, 那就錯了. FD_WRITE 事件只在傳送緩衝區有多出的空位, 可以容納需要傳送的資料時才會觸發.
上面所謂的傳送緩衝區,是指底層提供的緩衝區. send() 先將資料寫入到傳送緩衝區中, 然後透過傳送到接收端. 你或許會想, 只要不把傳送緩衝區填滿, 讓傳送緩衝區保持足夠多的空位容納需要傳送的資料, 那麼你就會源源不斷地收到 FD_WRITE 事件了. 嘿嘿, 錯了.上面只是說 FD_WRITE 事件在傳送緩衝區有多出的空位時會觸發, 但不是在有足夠的空位時觸發, 就是說你得先把傳送緩衝區填滿.
通常的辦法是在一個無限迴圈中不斷的傳送資料, 直到把傳送緩衝區填滿. 當傳送緩衝區被填滿後, send() 將會返回 SOCKET_ERROR , WSAGetLastError() 會返回 WSAWOULLOCK . 如果當前這個 SOCKET 處於阻塞(同步), 會一直等待直到傳送緩衝區空出位置然後傳送資料; 如果SOCKET是非阻塞(非同步)的,那麼你就會得到 WSAWOULDBLOCK 錯誤. 於是只要我們首先迴圈呼叫 send() 直到傳送緩衝區被填滿, 然後當緩衝區空出位置來的時候, 系統就會發出FD_WRITE事件. 有沒有想過我能指出這一點來是多麼不容易, 你可真走運. 下面是一個處理 FD_WRITE 事件的例子.
case FD_WRITE: // 可以傳送資料了
{
// 進入無限迴圈
while(TRUE)
{
// 從中讀取資料, 儲存到 packet.data 裡面.
in.read((char*)&packet.data, MAX_PACKET_SIZE);
// 傳送資料
if (send(wparam, (char*)(&packet), sizeof(PACKET), 0) == SOCKET_ERROR)
{
if (WSAGetLastError() == WSAEWOULDBLOCK)
{
// 傳送緩衝區已經滿了, 退出迴圈.
break;
}
else // 其他錯誤
{
// 顯示出錯資訊然後退出.
CleanUp();
return(0);
}
}
}
} break;
看到了吧, 實現其實一點也不困難. 你只是弄混了一些概念而已. 使用這樣的傳送方式, 在傳送緩衝區變滿的時候就可以退出迴圈. 然後, 當緩衝區空出位置來的時候, 系統會觸發另外一個 FD_WRITE 事件, 於是你就可以繼續傳送資料了.
在你開始使用新學到的知識之前, 我還想說明一下 FD_WRITE 事件的使用時機. 如果你不是一次性傳送大批次的資料的話, 就別想著使用 FD_WRITE 事件了, 原因很簡單 - 如果你寄期望於在收到 FD_WRITE 事件時傳送資料, 但是卻又不能傳送足夠的資料填滿傳送緩衝區, 那麼你就只能收到連線剛剛建立時觸發的那一次 FD_WRITE - 系統不會觸發更多的 FD_WRITE 了. 所以當你只是傳送儘可能少的資料的時候, 就忘掉 FD_WRITE 機制吧, 在任何你想傳送資料的時候直接呼叫 send() .
結論
這是我寫過的最長的一篇文章. 我也曾試圖儘可能把它寫短一些來吸引你的注意力, 但是有太多的內容要包括. 在剛剛使用非同步 SOCKET 時, 如果你沒有正確地理解它, 真的會把自己搞胡塗. 我希望我的文章教會了你如何使用它們. ___________________________________
這是我在 上搜到的一篇文章中的一部分. 雖然原作者的部分觀點似乎並不正確, 但是文章寫得很易懂. 其實, 如果你想收到 FD_WRITE 事件而你又無法先填滿傳送緩衝區, 可以呼叫 WSAAsync( ..., FD_WRITE ). 如果當前傳送緩衝區有空位, 系統會馬上給你發 FD_WRITE 事件.
FD_WRITE 訊息, MFC 的 CAsyncSocket 類將其對映為 OnSend() . FD_READ 訊息, 被對映為 OnReceive() 函式.
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/10752043/viewspace-998846/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- Socket程式設計中的同步、非同步、阻塞和非阻塞(轉)程式設計非同步
- Java網路程式設計--UDP傳送接收資料Java程式設計UDP
- SOCKET實現廣播(BoardCast)的傳送和接收 (轉)AST
- 玩轉 PHP 網路程式設計全套之資料接收與傳送PHP程式設計
- PHP傳送和接收POST資料PHP
- 043-socket程式設計傳送GET請求程式設計
- MPTCP 原始碼分析(四) 傳送和接收資料TCP原始碼
- 【測試】echo傳送和接收TCP/UDP資料包|shell 傳送TCP/UDP資料包TCPUDP
- 利用ASP傳送和接收XML資料的處理方法XML
- windows socket簡單使用--實現客戶端連結服務端併傳送和接收資料Windows客戶端服務端
- Go的Channel傳送和接收Go
- Socket程式設計注意接收緩衝區大小程式設計
- Linux網路程式設計--使用者資料包傳送(轉)Linux程式設計
- 使用postman傳送資料,springmvc接收資料的問題PostmanSpringMVC
- 使用python傳送和接收郵件Python
- 0229-UDP 傳送和接收UDP
- IO和socket程式設計程式設計
- 網路程式設計之 Udp接收資料程式設計UDP
- Socket 傳送檔案
- Java入門:UDP協議傳送/接收資料實現JavaUDP協議
- Java入門:TCP協議傳送/接收資料實現JavaTCP協議
- java傳送接收組播(多播)資料包(UDP包)JavaUDP
- http不使用Form表單傳送檔案資料和非檔案資料(上傳篇)HTTPORM
- 0230-TCP 傳送和接收TCP
- Python 網路程式設計-TCP服務端基礎篇(一)簡單的傳送和接收Python程式設計TCP服務端
- .NET非同步程式設計——給執行緒傳遞資料非同步程式設計執行緒
- UDP介紹及UDP傳送端和接收端廣播程式碼UDP
- java Socket接收資料亂碼問題Java
- Asp.net 利用Jquery Ajax傳送和接收DataTableASP.NETjQuery
- 使用python傳送郵件和接收郵件Python
- VC UDP接收 傳送 廣播UDP
- python 網路程式設計----非阻塞或非同步程式設計Python程式設計非同步
- [譯] 非同步程式設計:阻塞與非阻塞非同步程式設計
- PHP-Socket服務端客戶端傳送接收通訊例項詳解PHP服務端客戶端
- Udp廣播的傳送和接收(iOS + AsyncUdpSocket)下篇UDPiOS
- C# Socket 檔案傳送傳輸C#
- 非Access資料庫在VB的程式設計及應用 (轉)資料庫程式設計
- SOCKET程式設計程式設計