GetQueuedCompletionStatus函數(shù)有個(gè)OVERLAPPED結(jié)構(gòu),很多資料上都采用不同的結(jié)構(gòu)體來擴(kuò)展該結(jié)構(gòu),比如有的資料定義:
typedef struct _OVERLAPPEDPLUS
{
OVERLAPPED ol;
SOCKET s, sclient;
int OpCode;
WSABUF wbuf;
DWORD dwBytes, dwFlags;
}OVERLAPPEDPLUS;
然后,當(dāng)GetQueuedCompletionStatus(hIocp, &dwBytesXfered,(PULONG_PTR)&PerHandleKey, &Overlap, INFINITE);函數(shù)返回時(shí)候,人們常用OverlapPlus = CONTAINING_RECORD(Overlap, OVERLAPPEDPLUS, ol)得到一些信息。比如此時(shí)端口上完成的是什么操作,數(shù)據(jù)是什么等,還有,系統(tǒng)如何做到自動(dòng)填充上述的結(jié)構(gòu)的,也就是說,系統(tǒng)怎么知道在Overlap->OpCode存放的應(yīng)該是操作類型,如讀,寫操作,而在Overlap->wbuf存放的應(yīng)該是讀寫數(shù)據(jù)。
Overlap->OpCode,操作類型是在投遞WSASend,WSARecv的時(shí)候,由你自己指定填充這個(gè)字段。
因?yàn)槭欠嵌氯模扔谕哆f到與套接字相關(guān)聯(lián)的完成端口上,系統(tǒng)會(huì)把把WSASend對(duì)應(yīng)的緩沖區(qū)提交到底層緩沖,也可以把WSARecv投遞的緩沖區(qū),用接收到的數(shù)據(jù)填充,每一個(gè)WSASend,WSARecv,都應(yīng)有新申請(qǐng)一個(gè)overlaspped plus結(jié)構(gòu)提交,以存放本次投遞的IO操作的相關(guān)數(shù)據(jù),——單IO操作數(shù)據(jù)所以工作器線程中,從完成端口隊(duì)列中g(shù)et得到一個(gè)完成包的時(shí)候,可以根據(jù)單句柄數(shù)據(jù)知道在這個(gè)完成端口上是哪一個(gè)套接字投遞的IO操作完成了,從get到的overlapped中得到相關(guān)的已經(jīng)完成IO數(shù)據(jù)和信息,并作相應(yīng)的處理。比如投遞了1M,完成包卻告知只完成512K,那么你就知道要把余下的512K繼續(xù)投遞WSASend,當(dāng)然上一個(gè)WSASend的Overlapped這個(gè)時(shí)候可以重用到下一個(gè)WSASend中,這個(gè)是允許的,可以用一個(gè)字段存放全部1M,把余下未Send成功512k放到wbuf中,繼續(xù)投遞或者投遞WSARecv1M數(shù)據(jù),卻收到一個(gè)512K的完成通知,那么你要繼續(xù)投遞WSARecv,當(dāng)然前一個(gè)WSARecv的overlapped也可以重用,不過需要一些處理,把已經(jīng)接收到的512K保存到某個(gè)字段中,再投遞一個(gè)512K的請(qǐng)求去接收完成端口內(nèi)部,對(duì)投遞的Overlapped的填充,好像只有WSARecv的時(shí)候填充WSABUF,其他都是投遞IO前,代碼中顯式填充,并投遞的。至于完成了多少個(gè)字節(jié),是在lpNumberOfBytes中得到。
對(duì)GetQueuedCompletionStatus函數(shù)解釋:
實(shí)現(xiàn)從指定的IOCP獲取CP。當(dāng)CP隊(duì)列為空時(shí),對(duì)此函數(shù)的調(diào)用將被阻塞,而不是一直等待I/O的完成。當(dāng)CP隊(duì)列不為空時(shí),被阻塞的線程將以后進(jìn)先出(LIFO)順序被釋放。對(duì)于IOCP機(jī)制,它允許多線程并發(fā)調(diào)用GetQueuedCompletionStatus函數(shù),最大并發(fā)數(shù)是在調(diào)用CreateIoCompletionPort函數(shù)時(shí)指定的,超出最大并發(fā)數(shù)的調(diào)用線程,將被阻塞。函數(shù)解釋如下:
聲明:
BOOL GetQueuedCompletionStatus(
HANDLE CompletionPort,
LPDWORD lpNumberOfBytes,
PULONG_PTR lpCompletionKey,
LPOVERLAPPED *lpOverlapped,
DWORD dwMilliseconds);
調(diào)用參數(shù):
CompletionPort:指定的IOCP,該值由CreateIoCompletionPort函數(shù)創(chuàng)建。
lpnumberofbytes:一次完成后的I/O操作所傳送數(shù)據(jù)的字節(jié)數(shù)。
lpcompletionkey:當(dāng)文件I/O操作完成后,用于存放與之關(guān)聯(lián)的CK。
lpoverlapped:為調(diào)用IOCP機(jī)制所引用的OVERLAPPED結(jié)構(gòu)。
dwmilliseconds:用于指定調(diào)用者等待CP的時(shí)間。
返回值:
調(diào)用成功,則返回非零數(shù)值,相關(guān)數(shù)據(jù)存于lpNumberOfBytes、lpCompletionKey、lpCompletionKey變量中。失敗則返回零值。