• <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)崩潰。
            久久久久九九精品影院| 亚洲国产一成久久精品国产成人综合| 久久国产精品二国产精品| 亚洲人成网亚洲欧洲无码久久| 国产午夜福利精品久久| 久久精品国产免费一区| 久久精品www人人爽人人| 日韩精品久久久久久免费| 一本一道久久综合狠狠老| 亚洲一级Av无码毛片久久精品| 蜜桃麻豆www久久国产精品| 久久精品国产99久久久香蕉| 精品久久久久久无码中文字幕| 香蕉久久夜色精品国产小说| 日本福利片国产午夜久久| 精品久久久久久久久久久久久久久 | 国产福利电影一区二区三区久久老子无码午夜伦不 | 蜜臀av性久久久久蜜臀aⅴ| 久久中文字幕视频、最近更新| 精品久久久无码中文字幕| 久久人人爽人人爽人人片AV麻豆| 久久久WWW免费人成精品| 久久一本综合| 久久免费看黄a级毛片| 亚洲午夜无码久久久久| 精品少妇人妻av无码久久| 久久久青草久久久青草| 久久精品国产亚洲7777| 思思久久精品在热线热| 久久国产精品一国产精品金尊| 久久免费小视频| 综合久久一区二区三区 | 99久久免费国产精品热| 26uuu久久五月天| 7777精品伊人久久久大香线蕉 | 精品久久久久久综合日本| 精品久久国产一区二区三区香蕉 | 久久棈精品久久久久久噜噜| 亚洲国产二区三区久久| 久久九九久精品国产免费直播| 国内精品久久久人妻中文字幕|