• <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>

            笑看風(fēng)云淡

            寵辱不驚,看庭前花開(kāi)花落;去留無(wú)意,望天空云卷云舒
            posts - 96, comments - 48, trackbacks - 0, articles - 0
              C++博客 :: 首頁(yè) :: 新隨筆 ::  :: 聚合  :: 管理
            我本想把發(fā)送和接收分開(kāi)作為兩部分,但是最后我決定只略微解釋一下 FD_READ ,留下更多的時(shí)間來(lái)說(shuō)明更復(fù)雜的 FD_WRITE , FD_READ 事件非常容易掌握. 當(dāng)有數(shù)據(jù)發(fā)送過(guò)來(lái)時(shí), WinSock 會(huì)以 FD_READ 事件通知你, 對(duì)于每一個(gè) FD_READ 事件, 你需要像下面這樣調(diào)用 recv() :

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

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

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

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

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

            case FD_WRITE: // 可以發(fā)送數(shù)據(jù)了
            {
              // 進(jìn)入無(wú)限循環(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)滿(mǎn)了, 退出循環(huán).
                  break;
                }
                else // 其他錯(cuò)誤
                {
                  // 顯示出錯(cuò)信息然后退出.
                  CleanUp();
                  return(0);
                }
                }
              }
            } break;

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

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

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

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

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

            Feedback

            # re: 異步 SOCKET 編程 - 發(fā)送和接收數(shù)據(jù)  回復(fù)  更多評(píng)論   

            2012-05-24 23:39 by xx
            呵呵 謝謝分享

            # re: 異步 SOCKET 編程 - 發(fā)送和接收數(shù)據(jù)[未登錄](méi)  回復(fù)  更多評(píng)論   

            2012-05-25 14:08 by jianc
            學(xué)習(xí)ing

            只有注冊(cè)用戶(hù)登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問(wèn)   Chat2DB   管理


            久久婷婷人人澡人人爽人人爱| 久久99精品国产99久久6男男| 99久久国产亚洲高清观看2024| 国产亚洲精午夜久久久久久| 久久综合久久性久99毛片| 伊人久久大香线蕉综合Av | 一级做a爱片久久毛片| 国产精品久久久久久久久久影院| 国产三级久久久精品麻豆三级| 思思久久好好热精品国产| 国产精品成人久久久久三级午夜电影| 99精品久久精品一区二区| 99久久国产亚洲高清观看2024| 久久国产亚洲精品| 国产精品天天影视久久综合网| 亚洲伊人久久大香线蕉综合图片 | 亚洲国产成人精品91久久久 | 精品多毛少妇人妻AV免费久久| 国产精品国色综合久久| 欧美无乱码久久久免费午夜一区二区三区中文字幕 | 伊人色综合久久天天| 久久精品免费一区二区| 91久久成人免费| 精品久久久久久中文字幕人妻最新| 久久精品亚洲乱码伦伦中文| 国产精品VIDEOSSEX久久发布| 2021最新久久久视精品爱| 激情综合色综合久久综合| 久久se精品一区精品二区| 影音先锋女人AV鲁色资源网久久| 久久青青草原精品国产不卡| 欧美日韩中文字幕久久伊人| 国产一区二区精品久久凹凸| 国产亚洲精品自在久久| 久久婷婷五月综合色高清| 久久精品国产亚洲综合色| 色欲综合久久躁天天躁蜜桃| 久久精品免费一区二区三区| 99久久这里只有精品| 久久精品国产秦先生| 66精品综合久久久久久久|