• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            不會飛的鳥

            2010年12月10日 ... 不鳥他們!!! 我要用自己開發(fā)的分布式文件系統(tǒng)、分布式調(diào)度系統(tǒng)、分布式檢索系統(tǒng), 做自己的搜索引擎!!!大魚有大志!!! ---楊書童

            異步SOCKET編程-發(fā)送和接收數(shù)據(jù)[轉(zhuǎn)]

            我本想把發(fā)送和接收分開作為兩部分,但是最后我決定只略微解釋一下 FD_READ ,留下更多的時間來說明更復雜的 FD_WRITE , FD_READ 事件非常容易掌握. 當有數(shù)據(jù)發(fā)送過來時, WinSock 會以 FD_READ 事件通知你, 對于每一個 FD_READ 事件, 你需要像下面這樣調(diào)用 recv() :

            int bytes_recv = recv(wParam, &data, sizeof(data), 0);

            基本上就是這樣, 別忘了修改上面的 wParam. 還有, 不一定每一次調(diào)用 recv() 都會接收到一個完整的數(shù)據(jù)包, 因為數(shù)據(jù)可能不會一次性全部發(fā)送過來. 所以在開始處理接收到的數(shù)據(jù)之前, 最好對接收到的字節(jié)數(shù) ( 即 recv() 的返回值) 進行判斷, 看看是否收到的是一個完整的數(shù)據(jù)包.

            FD_WRITE 相對來說就麻煩一些. 首先, 當你建立了一個連接時, 會產(chǎn)生一個 FD_WRITE 事件. 但是如果你認為在收到 FD_WRITE 時調(diào)用 send() 就萬事大吉, 那就錯了. FD_WRITE 事件只在發(fā)送緩沖區(qū)有多出的空位, 可以容納需要發(fā)送的數(shù)據(jù)時才會觸發(fā).

            上面所謂的發(fā)送緩沖區(qū),是指系統(tǒng)底層提供的緩沖區(qū). send() 先將數(shù)據(jù)寫入到發(fā)送緩沖區(qū)中, 然后通過網(wǎng)絡(luò)發(fā)送到接收端. 你或許會想, 只要不把發(fā)送緩沖區(qū)填滿, 讓發(fā)送緩沖區(qū)保持足夠多的空位容納需要發(fā)送的數(shù)據(jù), 那么你就會源源不斷地收到 FD_WRITE 事件了. 嘿嘿, 錯了.上面只是說 FD_WRITE 事件在發(fā)送緩沖區(qū)有多出的空位時會觸發(fā), 但不是在有足夠的空位時觸發(fā), 就是說你得先把發(fā)送緩沖區(qū)填滿.

            通常的辦法是在一個無限循環(huán)中不斷的發(fā)送數(shù)據(jù), 直到把發(fā)送緩沖區(qū)填滿. 當發(fā)送緩沖區(qū)被填滿后, send() 將會返回 SOCKET_ERROR , WSAGetLastError() 會返回 WSAWOULDBLOCK . 如果當前這個 SOCKET 處于阻塞(同步)模式, 程序會一直等待直到發(fā)送緩沖區(qū)空出位置然后發(fā)送數(shù)據(jù); 如果SOCKET是非阻塞(異步)的,那么你就會得到 WSAWOULDBLOCK 錯誤. 于是只要我們首先循環(huán)調(diào)用 send() 直到發(fā)送緩沖區(qū)被填滿, 然后當緩沖區(qū)空出位置來的時候, 系統(tǒng)就會發(fā)出FD_WRITE事件. 有沒有想過我能指出這一點來是多么不容易, 你可真走運. 下面是一個處理 FD_WRITE 事件的例子.

            case FD_WRITE: // 可以發(fā)送數(shù)據(jù)了
            {
            // 進入無限循環(huán)
            while(TRUE)
            {
            // 從文件中讀取數(shù)據(jù), 保存到 packet.data 里面.
            in.read((char*)&packet.data, MAX_PACKET_SIZE);

            // 發(fā)送數(shù)據(jù)
            if (send(wparam, (char*)(&packet), sizeof(PACKET), 0) == SOCKET_ERROR)
            {
            if (WSAGetLastError() == WSAEWOULDBLOCK)
            {
            // 發(fā)送緩沖區(qū)已經(jīng)滿了, 退出循環(huán).
            break;
            }
            else // 其他錯誤
            {
            // 顯示出錯信息然后退出.
            CleanUp();
            return(0);
            }
            }
            }
            } break;

            看到了吧, 實現(xiàn)其實一點也不困難. 你只是弄混了一些概念而已. 使用這樣的發(fā)送方式, 在發(fā)送緩沖區(qū)變滿的時候就可以退出循環(huán). 然后, 當緩沖區(qū)空出位置來的時候, 系統(tǒng)會觸發(fā)另外一個 FD_WRITE 事件, 于是你就可以繼續(xù)發(fā)送數(shù)據(jù)了.

            在你開始使用新學到的知識之前, 我還想說明一下 FD_WRITE 事件的使用時機. 如果你不是一次性發(fā)送大批量的數(shù)據(jù)的話, 就別想著使用 FD_WRITE 事件了, 原因很簡單 - 如果你寄期望于在收到 FD_WRITE 事件時發(fā)送數(shù)據(jù), 但是卻又不能發(fā)送足夠的數(shù)據(jù)填滿發(fā)送緩沖區(qū), 那么你就只能收到連接剛剛建立時觸發(fā)的那一次 FD_WRITE - 系統(tǒng)不會觸發(fā)更多的 FD_WRITE 了. 所以當你只是發(fā)送盡可能少的數(shù)據(jù)的時候, 就忘掉 FD_WRITE 機制吧, 在任何你想發(fā)送數(shù)據(jù)的時候直接調(diào)用 send() .

            結(jié)論
            這是我寫過的最長的一篇文章. 我也曾試圖盡可能把它寫短一些來吸引你的注意力, 但是有太多的內(nèi)容要包括. 在剛剛使用異步SOCKET 時, 如果你沒有正確地理解它, 真的會把自己搞胡涂. 我希望我的文章教會了你如何使用它們. ___________________________________

            這是我在 GOOGLE 上搜到的一篇文章中的一部分. 雖然原作者的部分觀點似乎并不正確, 但是文章寫得很易懂. 其實, 如果你想收到 FD_WRITE事件而你又無法先填滿發(fā)送緩沖區(qū), 可以調(diào)用 WSAAsyncSelect( ..., FD_WRITE ). 如果當前發(fā)送緩沖區(qū)有空位, 系統(tǒng)會馬上給你發(fā) FD_WRITE 事件.

            FD_WRITE 消息, MFC 的 CAsyncSocket 類將其映射為 OnSend() 函數(shù). FD_READ 消息, 被映射為 OnReceive() 函數(shù).

            posted on 2006-04-29 10:08 不會飛的鳥 閱讀(4555) 評論(3)  編輯 收藏 引用

            評論

            # re: 異步SOCKET編程-發(fā)送和接收數(shù)據(jù)[轉(zhuǎn)] 2006-07-18 09:59 slanba

            很好的一篇文章,描述的很清晰,比較具體,  回復  更多評論   

            # re: 異步SOCKET編程-發(fā)送和接收數(shù)據(jù)[轉(zhuǎn)][未登錄] 2009-03-25 15:44 cc

            找了好久,終于找到一篇說得比較明白了文章
            謝謝  回復  更多評論   

            # re: 異步SOCKET編程-發(fā)送和接收數(shù)據(jù)[轉(zhuǎn)] 2010-02-13 16:12 阿長

            有一個疑問,您給的樣例代碼中:
            if (WSAGetLastError() == WSAEWOULDBLOCK)
            {
            // 發(fā)送緩沖區(qū)已經(jīng)滿了, 退出循環(huán).
            break;
            }
            那么退出循環(huán)以后,那么這個包是不是沒有正確發(fā)出去?
            那么再下一次發(fā)送的時候,發(fā)出去的將是下一個包,那么也就是說這個包就永遠的丟棄了?  回復  更多評論   

            中文字幕久久波多野结衣av| 欧美一级久久久久久久大| 中文国产成人精品久久不卡| 亚洲级αV无码毛片久久精品 | 99久久国产主播综合精品| 国产精品无码久久久久| 合区精品久久久中文字幕一区| 人妻少妇久久中文字幕| 国产999精品久久久久久| 久久久久久久97| 狠狠人妻久久久久久综合蜜桃| 国产成人久久精品一区二区三区| 国产精品久久久久久久久| 久久综合鬼色88久久精品综合自在自线噜噜 | 熟妇人妻久久中文字幕| 久久久国产精华液| 久久久精品午夜免费不卡| 18禁黄久久久AAA片| 狠狠久久综合伊人不卡| 精品久久无码中文字幕| 久久人妻无码中文字幕| 日批日出水久久亚洲精品tv| 国产精品久久成人影院| 亚洲AV无一区二区三区久久| 伊色综合久久之综合久久| 精品国产乱码久久久久久浪潮| 999久久久免费精品国产| 亚洲∧v久久久无码精品| 久久精品中文字幕一区| 日韩美女18网站久久精品| 久久国产福利免费| 99久久婷婷国产综合精品草原| 国内精品久久久久久野外| 午夜精品久久久久久久| 伊人久久大香线焦AV综合影院| 一级做a爰片久久毛片免费陪| 四虎国产精品成人免费久久| 亚洲一区精品伊人久久伊人 | 国产精品视频久久久| 欧美伊香蕉久久综合类网站| 精品精品国产自在久久高清|