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

            S.l.e!ep.¢%

            像打了激速一樣,以四倍的速度運(yùn)轉(zhuǎn),開心的工作
            簡單、開放、平等的公司文化;尊重個(gè)性、自由與個(gè)人價(jià)值;
            posts - 1098, comments - 335, trackbacks - 0, articles - 1
              C++博客 :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理
            關(guān)于Socket和IOCP的一些值得注意的地方收藏

            ??? IOCP是一整套高性能的IO操作異步模型,可以用在文件操作也可以用在網(wǎng)絡(luò)SOCKET操作上面。當(dāng)用在網(wǎng)絡(luò)SOCKET上時(shí),在服務(wù)器端主要配合AceeptEx WSASend WSASendto來使用,在客戶機(jī)端主要配合ConnectEx WSARecv和WSARecvFrom來使用。這幾天用IOCP模型模仿IPMSG軟件時(shí)有一些感觸,分享如下:(這里沒有具體的使用常識(shí),這部分請(qǐng)參考《Windows網(wǎng)絡(luò)編程2nd》或者相關(guān)網(wǎng)路資料)

            ?

            一、單句柄數(shù)據(jù)和單IO數(shù)據(jù)

            ????? 這部分的術(shù)語不是很明白如何而來,只是根據(jù)Windows網(wǎng)絡(luò)編程一書的中文翻譯而來。

            ????? 單句柄數(shù)據(jù)是跟隨你丟給IOCP的相關(guān)句柄的,而IO數(shù)據(jù)則是根據(jù)你每次IO操作時(shí)丟給相關(guān)API函數(shù)的OVERLAPPED參數(shù)的指針。具體來說,如果你要把某個(gè)句柄上的操作用IOCP來完成,那么你會(huì)調(diào)用一次(注意,僅需一次,以前我會(huì)在每次IO操作時(shí)丟調(diào)用,這是錯(cuò)誤的示范!)CreateIoCompletionPort時(shí)把他的指針賦值給CompletionKey這個(gè)參數(shù),而這塊堆上內(nèi)存將會(huì)跟隨你的句柄直到句柄被Close,而且中間不允許更換,所以說單句柄數(shù)據(jù)應(yīng)該而且必須是與你的IO句柄相關(guān)的數(shù)據(jù)比如說socket跟狀態(tài)等等。

            ????? 而單IO數(shù)據(jù)是在調(diào)用WSARecv等等的API函數(shù)時(shí)的OVERLAPPED參數(shù)指向的堆上內(nèi)存,這部分的數(shù)據(jù)結(jié)構(gòu)最簡單的做法是把OVERLAPPED作為數(shù)據(jù)結(jié)構(gòu)的第一個(gè)字段,然后后面跟上跟此次IO操作相關(guān)的一些數(shù)據(jù),比如說指向緩沖區(qū)的指針和表明緩沖區(qū)長度的DWORD值等等。這部分的數(shù)據(jù)只跟每次調(diào)用API函數(shù)進(jìn)行的IO操作相關(guān)。

            二、AcceptEx函數(shù)

            ????? 我在這個(gè)函數(shù)上卡殼了很長時(shí)間,他第三個(gè)函數(shù)表示一個(gè)完成AcceptEx操作后用來接收數(shù)據(jù)的一個(gè)緩沖,第四個(gè)參數(shù)表示一個(gè)緩沖的大小,然后四個(gè)函數(shù)分別表示本地、遠(yuǎn)程地址結(jié)構(gòu)的長度。如果你只想做Accept操作而不想在這里做接收數(shù)據(jù)的動(dòng)作那么把第四個(gè)參數(shù)設(shè)為0即可。但是容易在這里犯錯(cuò)的是,如果你認(rèn)為既然不要接收數(shù)據(jù)那么把第三個(gè)參數(shù)設(shè)定為NULL那么這次投遞永遠(yuǎn)不可能完成,并且所有的返回值WSAGetLastError都會(huì)看上去非常正確,這很不幸。即使你不想接收任何數(shù)據(jù)你也不能把表示緩沖區(qū)的參數(shù)設(shè)為0,而要至少設(shè)置一個(gè)長度為兩個(gè)地址結(jié)構(gòu)長度加上32的長度才行,如果不到那個(gè)長度那么等著在delete的時(shí)候報(bào)運(yùn)行時(shí)錯(cuò)誤吧!后面兩個(gè)表示地址結(jié)構(gòu)長度的參數(shù)都必須設(shè)置成地址結(jié)構(gòu)長度加上16字節(jié)。如果你打算從緩沖里取出那兩個(gè)地址結(jié)構(gòu),那么切記在每個(gè)地址結(jié)構(gòu)后面都有16字節(jié)的數(shù)據(jù)塊,這兩塊數(shù)據(jù)到底是什么我也不知道,也沒有任何資料給我解釋包括MSDN,相當(dāng)崩潰!

            三、ConnectEx函數(shù)

            ????? 基本上這個(gè)函數(shù)至少從表面上沒有AcceptEx函數(shù)那些龜毛和詭異的東西,但是你認(rèn)為這跟WSARecv之類API一樣直接簡單你就又錯(cuò)了。你會(huì)發(fā)現(xiàn)按照普通的方法調(diào)用以后調(diào)用WSAGetLastError返回的是10022錯(cuò)誤,而不是WSA_IO_PENDING,又崩潰了吧?還好,這次MSDN給了你一小行解釋,說The parameter s is an unbound or a listening socket,還是詭異兩個(gè)字connect操作干嘛要綁定?不知道,沒人給解釋,那綁定就對(duì)了,那么綁哪個(gè)?最好把你的地址結(jié)構(gòu)像下面這樣設(shè)置

            SOCKADDR_IN temp;

            temp.sin_family = AF_INET;

            temp.sin_port = htons(0);

            temp.sin_addr.s_addr = htonl(ADDR_ANY);

            為什么端口這個(gè)地方用0,原因很簡單,你去查查MSDN,這樣表示他會(huì)在1000-4000這個(gè)范圍(可能記錯(cuò),想了解的話去查MSDN)找一個(gè)沒被用到的port,這樣的話最大程度保證你bind的成功,然后再把socket句柄丟給IOCP,然后調(diào)用ConnectEx這樣就會(huì)看到熟悉的WSA_IO_PENDING了!

            四、WSARecvFrom和WSASendTo

            ????? 這兩個(gè)函數(shù)沒什么詭異的地方,只有一個(gè)細(xì)節(jié),由于這兩個(gè)函數(shù)都是在UDP里用,所以有個(gè)地址結(jié)構(gòu)參數(shù),WSARecvFrom的地址結(jié)構(gòu)API會(huì)自己抓取可以在堆棧上分配,而WSASendTo的地址結(jié)構(gòu)API不會(huì)自己抓取所以需要你用new在堆上分配,在完成以后再delete掉。

            ????? 另外還有就是基于UDP的IOCP在WIN2K上可能有些問題,這個(gè)在google大神上很容易找到,比如說你打個(gè)WSARecvFrom就能在第一頁看到,在WINXP上則沒有什么問題。

            ?

            ????? 仔細(xì)玩了兩天IOCP以后發(fā)現(xiàn),細(xì)節(jié)很重要,無論是看書還是MSDN等等英文資料,不要錯(cuò)過任何一個(gè)單詞,每錯(cuò)過一個(gè)單詞就多一個(gè)可能讓你在某個(gè)地方多調(diào)試一個(gè)小時(shí)甚至更多~

            Feedback

            # re: 關(guān)于Socket和IOCP的一些值得注意的地方  回復(fù)  更多評(píng)論   

            2009-05-21 14:06 by wz19860913
            我認(rèn)為還有一個(gè)值得注意的地方,那就是異常斷開連接時(shí)的處理。
            在異常斷開連接時(shí),比如客戶端突然斷線等,服務(wù)器必須要謹(jǐn)慎編寫清理資源的代碼,否則服務(wù)器容易導(dǎo)致內(nèi)存泄漏或系統(tǒng)崩潰。
            性欧美丰满熟妇XXXX性久久久| 久久婷婷五月综合97色| 青青草原综合久久| 国产一区二区三区久久| 久久99精品久久久久久hb无码| 99久久人妻无码精品系列| 青青青国产精品国产精品久久久久| 国产精品久久久天天影视| 久久国产精品-久久精品| 国产成人精品久久亚洲高清不卡| 国产精品九九久久免费视频 | 狠狠久久综合伊人不卡| 久久伊人中文无码| 色偷偷久久一区二区三区| 亚洲成人精品久久| 亚洲精品WWW久久久久久| 亚洲国产精品高清久久久| 亚洲精品高清久久| 狠狠色婷婷久久一区二区| 久久综合久久久| 久久精品极品盛宴观看| 国产韩国精品一区二区三区久久| 欧洲国产伦久久久久久久| 无码人妻精品一区二区三区久久久| 国产亚州精品女人久久久久久 | 久久婷婷人人澡人人| 久久久一本精品99久久精品66| 精品乱码久久久久久夜夜嗨| 亚洲AV无码久久精品蜜桃| 亚洲?V乱码久久精品蜜桃 | 久久久久99这里有精品10 | 久久99热这里只有精品66| 国产一区二区三区久久精品| 亚洲国产另类久久久精品| 精品久久久久久无码人妻蜜桃| 国产精品禁18久久久夂久| 久久久久久久综合狠狠综合| 久久久久九九精品影院| 66精品综合久久久久久久| 99久久精品国产高清一区二区| 丁香色欲久久久久久综合网|