1. 程式人生 > >【轉】關於Socket和IOCP的一些值得注意的地方

【轉】關於Socket和IOCP的一些值得注意的地方

轉至:http://laokaddk.blog.51cto.com/368606/287039
關於Socket和IOCP的一些值得注意的地方 


IOCP是一整套高效能的IO操作非同步模型,可以用在檔案操作也可以用在網路SOCKET操作上面。當用在網路SOCKET上時,在伺服器端主要配合AceeptEx WSASend WSASendto來使用,在客戶機端主要配合ConnectEx WSARecv和WSARecvFrom來使用。這幾天用IOCP模型模仿IPMSG軟體時有一些感觸,分享如下:(這裡沒有具體的使用常識,這部分請參考《Windows網路程式設計2nd》或者相關網路資料)



一、單控制代碼資料和單IO資料


這部分的術語不是很明白如何而來,只是根據Windows網路程式設計一書的中文翻譯而來。

單控制代碼資料是跟隨你丟給IOCP的相關控制代碼的,而IO資料則是根據你每次IO操作時丟給相關API函式的OVERLAPPED引數的指標。具體來說,如果你要把某個控制代碼上的操作用IOCP來完成,那麼你會呼叫一次(注意,僅需一次,以前我會在每次IO操作時丟呼叫,這是錯誤的示範!)CreateIoCompletionPort時把他的指標賦值給CompletionKey這個引數,而這塊堆上記憶體將會跟隨你的控制代碼直到控制代碼被Close,而且中間不允許更換,所以說單控制代碼資料應該而且必須是與你的IO控制代碼相關的資料比如說socket跟狀態等等。

而單IO資料是在呼叫WSARecv等等的API函式時的OVERLAPPED引數指向的堆上記憶體,這部分的資料結構最簡單的做法是把OVERLAPPED作為資料結構的第一個欄位,然後後面跟上跟此次IO操作相關的一些資料,比如說指向緩衝區的指標和表明緩衝區長度的DWORD值等等。這部分的資料只跟每次呼叫API函式進行的IO操作相關。

二、AcceptEx函式


我在這個函式上卡殼了很長時間,他第三個函式表示一個完成AcceptEx操作後用來接收資料的一個緩衝,第四個引數表示一個緩衝的大小,然後四個函式分別表示本地、遠端地址結構的長度。如果你只想做Accept操作而不想在這裡做接收資料的動作那麼把第四個引數設為0即可。但是容易在這裡犯錯的是,如果你認為既然不要接收資料那麼把第三個引數設定為NULL那麼這次投遞永遠不可能完成,並且所有的返回值WSAGetLastError都會看上去非常正確,這很不幸。即使你不想接收任何資料你也不能把表示緩衝區的引數設為0,而要至少設定一個長度為兩個地址結構長度加上32的長度才行,如果不到那個長度那麼等著在delete的時候報執行時錯誤吧!後面兩個表示地址結構長度的引數都必須設定成地址結構長度加上16位元組。如果你打算從緩衝裡取出那兩個地址結構,那麼切記在每個地址結構後面都有16位元組的資料塊,這兩塊資料到底是什麼我也不知道,也沒有任何資料給我解釋包括MSDN,相當崩潰!

三、ConnectEx函式


基本上這個函式至少從表面上沒有AcceptEx函式那些龜毛和詭異的東西,但是你認為這跟WSARecv之類API一樣直接簡單你就又錯了。你會發現按照普通的方法呼叫以後呼叫WSAGetLastError返回的是10022錯誤,而不是WSA_IO_PENDING,又崩潰了吧?還好,這次MSDN給了你一小行解釋,說The parameter s is an unbound or a listening socket,還是詭異兩個字connect操作幹嘛要繫結?不知道,沒人給解釋,那繫結就對了,那麼綁哪個?最好把你的地址結構像下面這樣設定

SOCKADDR_IN temp;

temp.sin_family = AF_INET;

temp.sin_port = htons(0);

temp.sin_addr.s_addr = htonl(ADDR_ANY);

為什麼埠這個地方用0,原因很簡單,你去查查MSDN,這樣表示他會在1000-4000這個範圍(可能記錯,想了解的話去查MSDN)找一個沒被用到的port,這樣的話最大程度保證你bind的成功,然後再把socket控制代碼丟給IOCP,然後呼叫ConnectEx這樣就會看到熟悉的WSA_IO_PENDING了!

四、WSARecvFrom和WSASendTo

這兩個函式沒什麼詭異的地方,只有一個細節,由於這兩個函式都是在UDP裡用,所以有個地址結構引數,WSARecvFrom的地址結構API會自己抓取可以在堆疊上分配,而WSASendTo的地址結構API不會自己抓取所以需要你用new在堆上分配,在完成以後再delete掉。

另外還有就是基於UDP的IOCP在WIN2K上可能

有些問題,這個在google大神上很容易找到,比如說你打個WSARecvFrom就能在第一頁看到,在WINXP上則沒有什麼問題。




仔細玩了兩天IOCP以後發現,細節很重要,無論是看書還是MSDN等等英文資料,不要錯過任何一個單詞,每錯過一個單詞就多一個可能讓你在某個地方多除錯一個小時甚至更多~
<script>window._bd_share_config={"common":{"bdSnsKey":{},"bdText":"","bdMini":"2","bdMiniList":false,"bdPic":"","bdStyle":"0","bdSize":"16"},"share":{}};with(document)0[(getElementsByTagName('head')[0]||body).appendChild(createElement('script')).src='http://bdimg.share.baidu.com/static/api/js/share.js?v=89860593.js?cdnversion='+~(-new Date()/36e5)];</script> 閱讀(375) | 評論(0) | 轉發(0) | 給主人留下些什麼吧!~~ 評論熱議