?
?
鄧立波 深圳,2007-8
作者聯(lián)系方式
:
email:????????
libodeng@gmail.com
msn:? ????????
libodeng@gmail.com
tel:???? ???????? 13510275799
版權(quán)/著作權(quán)所有 (C) 2007 鄧立波 保留所有權(quán)利
警告:未經(jīng)作者許可,任何人或組織不得轉(zhuǎn)載,公開發(fā)布,拷貝,傳播本文獻(xiàn)的全部或部分
?
誤區(qū)一,使用
send
函數(shù)發(fā)送數(shù)據(jù)
??????
一些人使用阻塞的
send
函數(shù)發(fā)送數(shù)據(jù),這是絕對應(yīng)該避免的,一旦某一個(gè)連接傳輸發(fā)生擁塞,或者突然中斷而沒有通知,調(diào)用
send
函數(shù)的線程將可能被阻塞很長一段時(shí)間(可能
10
秒或更長),尤其是當(dāng)服務(wù)器同時(shí)處理成千上萬個(gè)連接時(shí),這種情況可能會(huì)頻繁出現(xiàn)。
??????
使用非阻塞的
send
也不適合,數(shù)據(jù)不一定每次都能完全發(fā)送出去,你得使用
select
跟蹤這些
socket
的發(fā)送狀態(tài)(一旦可能,發(fā)送剩余數(shù)據(jù)),這失去了
IOCP
的優(yōu)勢。
?
誤區(qū)二,在多個(gè)工作線程的情況下使用
PostQueuedCompletionStatus
??????
使用
PostQueuedCompletionStatus
向
IOCP
發(fā)送一個(gè)“寫數(shù)據(jù)
IO
包”,然后由
IOCP
在工作線程里面調(diào)用
WSASend
發(fā)送數(shù)據(jù),這在單個(gè)工作線程存在的情況下是安全的,這些“寫數(shù)據(jù)
IO
包”將按它們被投遞的順序取出,并在工作線程中逐一處理。但在多個(gè)工作線程存在的情況下,數(shù)據(jù)仍然按它們投遞的順序取出,但處理不一定有序(即
WSASend
不一定按
PostQueuedCompletionStatus
投遞的順序被調(diào)用,iocp是個(gè)嚴(yán)格的fifo,線程切換卻是隨機(jī)的),而導(dǎo)致對方接收到的數(shù)據(jù)可能出現(xiàn)亂序。
?
?
比較好的一種方式是使用調(diào)用
WSASend
直接發(fā)送數(shù)據(jù),注意對同一個(gè)連接連續(xù)調(diào)用多次
WSASend
是安全的,而不需要等到前一個(gè)
WSASend
的操作完成,數(shù)據(jù)將按調(diào)用
WSASend
的順序發(fā)送。
在大多數(shù)情況下,上面的方式都可以滿足要求,但對一個(gè)大容量和高并發(fā)的服務(wù)器,可能還需要對同一個(gè)連接限制并發(fā)的
IO
次數(shù),以避免沖擊可分頁內(nèi)存鎖定極限和非分頁內(nèi)存極限,更好的方式是對每個(gè)連接僅允許一個(gè)
pending send IO
,這可以通過建一個(gè)發(fā)送緩存隊(duì)列實(shí)現(xiàn),當(dāng)
WSASend
未返回操作完成時(shí),后續(xù)提交的數(shù)據(jù)將放置于發(fā)送隊(duì)列,直到
WSASend
完成,然后從發(fā)送隊(duì)列取出一塊數(shù)據(jù),繼續(xù)發(fā)送。