Posted on 2008-03-12 11:51
小明 閱讀(6974)
評論(7) 編輯 收藏 引用 所屬分類:
Win32 、
Network/ACE
Tip 1 : 使用WSASend/WSARecv來收發數據,而不是使用ReadFile/WriteFile
一句話,前者具有更好的性能
Tip 2: 理解IOCP的最大并發線程數和工作線程數
應該讓工作線程數(調用GetQueuedCompletionStatus那些線程)大于等于在CreateIoCompletionPort 指定的NumberOfConcurrentThreads數。
標準做法是永遠設置NumberOfConcurrentThreads=0
Tip 3: 利用GetQueuedCompletionStatus的completion key和overlapped structure參數在異步操作中來傳遞信息
通常completion key用來傳遞和handle/socket/session的信息
而overlapped structure用來傳遞每次異步I/O的一些信息,通常的做法是會定義一個structure來派生于OVERLAPPED
struct MY_IO_DATA : public OVERLAPPED
Tip 4: 理解IOCP的完成包的排隊行為
從GetQueuedCompletionStatus得到完成包的次序可能跟調用WSASend/WSARecv的次序不一樣。
微軟唯一保證是如果調用WSASend/WSARecv得到SUCCESS或者IO_PENDING,就一定會有一個完成包出現在IOCP的隊列上,不管這個socket是否關閉了。
如果關閉socket,那么之后的WSASend/WSARecv調用就一定返回失敗的結果。
關于IOCP包可能次序錯亂和解決方法,有一篇文章可以參考: http://www.codeproject.com/KB/IP/reusablesocketserver4.aspx
我的做法是避免多次調用WSARecv
Tip 5: IOCP的清除
最重要的一點是,在I/O完成之前,不要釋放overlapped structure。可以用HasOverlappedIoCompleted來監測OV是否完成。
通常的做法是
1) 調用PostQueueCompletionStatus N次(N=工作線程數),來傳遞特殊的退出信息給所有的工作線程
2) 關閉所有的socket,如果很在意處理完未完成的數據包,需要使用一個計數器來跟蹤異步I/O事件,直到計數器為0,才關閉相應的socket
3) 關閉completion port