青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

posts - 126,  comments - 73,  trackbacks - 0

Windows完成端口編程


目錄
一 基本概念
二 OVERLAPPED數據結構
三 完成端口的內部機制
創建完成端口
完成端口線程的工作原理
線程間數據傳遞
線程的安全退出

一 基本概念
?????? 設備---windows操作系統上允許通信的任何東西,比如文件、目錄、串行口、并行

口、郵件槽、命名管道、無名管道、套接字、控制臺、邏輯磁盤、物理磁盤等。絕大多數

與設備打交道的函數都是CreateFile/ReadFile/WriteFile等。所以我們不能看到**File函

數就只想到文件設備。

?????? 與設備通信有兩種方式,同步方式和異步方式。同步方式下,當調用ReadFile函數

時,函數會等待系統執行完所要求的工作,然后才返回;異步方式下,ReadFile這類函數

會直接返回,系統自己去完成對設備的操作,然后以某種方式通知完成操作。

?????? 重疊I/O----顧名思義,當你調用了某個函數(比如ReadFile)就立刻返回做自己

的其他動作的時候,同時系統也在對I/0設備進行你要求的操作,在這段時間內你的程序和

系統的內部動作是重疊的,因此有更好的性能。所以,重疊I/O是用于異步方式下使用I/O

設備的。

重疊I/O需要使用的一個非常重要的數據結構OVERLAPPED。

?????? 完成端口---是一種WINDOWS內核對象。完成端口用于異步方式的重疊I/0情況下,

當然重疊I/O不一定非使用完成端口不可,還有設備內核對象、事件對象、告警I/0等。但

是完成端口內部提供了線程池的管理,可以避免反復創建線程的開銷,同時可以根據CPU的

個數靈活的決定線程個數,而且可以讓減少線程調度的次數從而提高性能。


二 OVERLAPPED數據結構
typedef struct _OVERLAPPED {

????ULONG_PTR Internal;//被系統內部賦值,用來表示系統狀態

????ULONG_PTR InternalHigh;// 被系統內部賦值,傳輸的字節數

????union {

????????struct {

????????????DWORD Offset;//和OffsetHigh合成一個64位的整數,用來表示從文件頭部的

多少字節開始

????????????DWORD OffsetHigh;//操作,如果不是對文件I/O來操作,則必須設定為0

????????};

????????PVOID Pointer;

????};

????HANDLE??hEvent;//如果不使用,就務必設為0,否則請賦一個有效的Event句柄

} OVERLAPPED, *LPOVERLAPPED;


下面是異步方式使用ReadFile的一個例子

OVERLAPPED Overlapped;

Overlapped.Offset=345;

Overlapped.OffsetHigh=0;

Overlapped.hEvent=0;

//假定其他參數都已經被初始化

ReadFile(hFile,buffer,sizeof(buffer),&dwNumBytesRead,&Overlapped);

這樣就完成了異步方式讀文件的操作,然后ReadFile函數返回,由操作系統做自己的事情



下面介紹幾個與OVERLAPPED結構相關的函數

等待重疊I/0操作完成的函數

BOOL GetOverlappedResult (

HANDLE hFile,

LPOVERLAPPED lpOverlapped,//接受返回的重疊I/0結構

LPDWORD lpcbTransfer,//成功傳輸了多少字節數

BOOL fWait //TRUE只有當操作完成才返回,FALSE直接返回,如果操作沒有完成,通過

調//用GetLastError ( )函數會返回ERROR_IO_INCOMPLETE

);

宏HasOverlappedIoCompleted可以幫助我們測試重疊I/0操作是否完成,該宏對OVERLAPPED

結構的Internal成員進行了測試,查看是否等于STATUS_PENDING值。

三 完成端口的內部機制
創建完成端口

?????? 完成端口是一個內核對象,使用時他總是要和至少一個有效的設備句柄進行關聯,

完成端口是一個復雜的內核對象,創建它的函數是:

HANDLE CreateIoCompletionPort(

????IN HANDLE FileHandle,

????IN HANDLE ExistingCompletionPort,

????IN ULONG_PTR CompletionKey,

????IN DWORD NumberOfConcurrentThreads

????);

通常創建工作分兩步:

第一步,創建一個新的完成端口內核對象,可以使用下面的函數:

?????? HANDLE CreateNewCompletionPort(DWORD dwNumberOfThreads)

{

??????????return CreateIoCompletionPort

(INVALID_HANDLE_VALUE,NULL,NULL,dwNumberOfThreads);

};

第二步,將剛創建的完成端口和一個有效的設備句柄關聯起來,可以使用下面的函數:

?????? bool AssicoateDeviceWithCompletionPort(HANDLE hCompPort,HANDLE

hDevice,DWORD dwCompKey)

{

??????????HANDLE h=CreateIoCompletionPort(hDevice,hCompPort,dwCompKey,0);

??????????return h==hCompPort;

};

說明

1)??CreateIoCompletionPort函數也可以一次性的既創建完成端口對象,又關聯到一個有

效的設備句柄

2)??CompletionKey是一個可以自己定義的參數,我們可以把一個結構的地址賦給它,然

后在合適的時候取出來使用,最好要保證結構里面的內存不是分配在棧上,除非你有十分

的把握內存會保留到你要使用的那一刻。

3)??NumberOfConcurrentThreads通常用來指定要允許同時運行的的線程的最大個數。通

常我們指定為0,這樣系統會根據CPU的個數來自動確定。

創建和關聯的動作完成后,系統會將完成端口關聯的設備句柄、完成鍵作為一條紀錄加入

到這個完成端口的設備列表中。如果你有多個完成端口,就會有多個對應的設備列表。如

果設備句柄被關閉,則表中自動刪除該紀錄。


完成端口線程的工作原理

?????? 完成端口可以幫助我們管理線程池,但是線程池中的線程需要我們使用

_beginthreadex來創建,憑什么通知完成端口管理我們的新線程呢?答案在函數

GetQueuedCompletionStatus。該函數原型:



BOOL GetQueuedCompletionStatus(

????IN??HANDLE CompletionPort,

????OUT LPDWORD lpNumberOfBytesTransferred,

????OUT PULONG_PTR lpCompletionKey,

????OUT LPOVERLAPPED *lpOverlapped,

????IN??DWORD dwMilliseconds

);


這個函數試圖從指定的完成端口的I/0完成隊列中抽取紀錄。只有當重疊I/O動作完成的時

候,完成隊列中才有紀錄。凡是調用這個函數的線程將被放入到完成端口的等待線程隊列

中,因此完成端口就可以在自己的線程池中幫助我們維護這個線程。

完成端口的I/0完成隊列中存放了當重疊I/0完成的結果---- 一條紀錄,該紀錄擁有四個字

段,前三項就對應GetQueuedCompletionStatus函數的2、3、4參數,最后一個字段是錯誤

信息dwError。我們也可以通過調用PostQueudCompletionStatus模擬完成了一個重疊I/0操

作。

當I/0完成隊列中出現了紀錄,完成端口將會檢查等待線程隊列,該隊列中的線程都是通過

調用GetQueuedCompletionStatus函數使自己加入隊列的。等待線程隊列很簡單,只是保存

了這些線程的ID。完成端口會按照后進先出的原則將一個線程隊列的ID放入到釋放線程列

表中,同時該線程將從等待GetQueuedCompletionStatus函數返回的睡眠狀態中變為可調度

狀態等待CPU的調度。

基本上情況就是如此,所以我們的線程要想成為完成端口管理的線程,就必須要調用

GetQueuedCompletionStatus函數。出于性能的優化,實際上完成端口還維護了一個暫停線

程列表,具體細節可以參考《Windows高級編程指南》,我們現在知道的知識,已經足夠了



線程間數據傳遞

?????? 線程間傳遞數據最常用的辦法是在_beginthreadex函數中將參數傳遞給線程函數,

或者使用全局變量。但是完成端口還有自己的傳遞數據的方法,答案就在于CompletionKey

和OVERLAPPED參數。

CompletionKey被保存在完成端口的設備表中,是和設備句柄一一對應的,我們可以將與設

備句柄相關的數據保存到CompletionKey中,或者將CompletionKey表示為結構指針,這樣

就可以傳遞更加豐富的內容。這些內容只能在一開始關聯完成端口和設備句柄的時候做,

因此不能在以后動態改變。

OVERLAPPED參數是在每次調用ReadFile這樣的支持重疊I/0的函數時傳遞給完成端口的。我

們可以看到,如果我們不是對文件設備做操作,該結構的成員變量就對我們幾乎毫無作用

。我們需要附加信息,可以創建自己的結構,然后將OVERLAPPED結構變量作為我們結構變

量的第一個成員,然后傳遞第一個成員變量的地址給ReadFile函數。因為類型匹配,當然

可以通過編譯。當GetQueuedCompletionStatus函數返回時,我們可以獲取到第一個成員變

量的地址,然后一個簡單的強制轉換,我們就可以把它當作完整的自定義結構的指針使用

,這樣就可以傳遞很多附加的數據了。太好了!只有一點要注意,如果跨線程傳遞,請注

意將數據分配到堆上,并且接收端應該將數據用完后釋放。我們通常需要將ReadFile這樣

的異步函數的所需要的緩沖區放到我們自定義的結構中,這樣當

GetQueuedCompletionStatus被返回時,我們的自定義結構的緩沖區變量中就存放了I/0操

作的數據。

CompletionKey和OVERLAPPED參數,都可以通過GetQueuedCompletionStatus函數獲得。

線程的安全退出

?????? 很多線程為了不止一次的執行異步數據處理,需要使用如下語句

while (true)

{

?????? .。。。。。。

?????? GetQueuedCompletionStatus(...);



??????????????。。。。。。

}

那么如何退出呢,答案就在于上面曾提到的PostQueudCompletionStatus函數,我們可以用

它發送一個自定義的包含了OVERLAPPED成員變量的結構地址,里面包含一個狀態變量,當

狀態變量為退出標志時,線程就執行清除動作然后退出。





FROM:http://www.vchelp.net/cndevforum/subject_view.asp?subject_id=176818&forum_id=55
posted on 2007-01-31 14:10 我風 閱讀(488) 評論(0)  編輯 收藏 引用

只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


<2007年8月>
2930311234
567891011
12131415161718
19202122232425
2627282930311
2345678

常用鏈接

留言簿(12)

隨筆分類

隨筆檔案

文章檔案

相冊

收藏夾

C++

MyFavorite

搜索

  •  

積分與排名

  • 積分 - 329312
  • 排名 - 76

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            国产精品视频一二三| 性久久久久久久| 欧美亚洲三区| 亚洲欧美变态国产另类| 欧美成人国产一区二区| 久久精品一区二区三区不卡牛牛 | 免费在线亚洲| 国产日韩亚洲欧美| 亚洲永久免费精品| 一区二区三区产品免费精品久久75| 久久激情视频免费观看| 欧美在线观看一区| 国产精品免费看片| 在线亚洲一区观看| 99ri日韩精品视频| 欧美激情精品久久久久久| 老司机精品导航| 国产一区二区三区四区五区美女 | 校园春色国产精品| 欧美系列亚洲系列| 一区二区三区欧美亚洲| 亚洲视频在线观看三级| 欧美色大人视频| 99v久久综合狠狠综合久久| 亚洲美女av黄| 欧美伦理91| 一区二区三欧美| 亚洲欧美国产制服动漫| 国产精品美女www爽爽爽视频| 一本色道久久综合亚洲精品高清 | 免费不卡在线视频| 伊人久久大香线蕉av超碰演员| 久久www免费人成看片高清| 久久久久成人精品免费播放动漫| 国产午夜精品理论片a级大结局| 午夜宅男久久久| 久久久久久夜精品精品免费| 国内揄拍国内精品少妇国语| 麻豆成人91精品二区三区| 亚洲国产日韩在线| 日韩亚洲精品电影| 国产精品毛片一区二区三区| 亚洲永久免费| 免费观看30秒视频久久| 亚洲精品乱码久久久久久日本蜜臀| 欧美激情综合亚洲一二区| 亚洲精品欧美日韩专区| 亚洲欧美日韩国产成人精品影院| 国产欧美视频一区二区| 另类激情亚洲| 中国成人黄色视屏| 久久久久99精品国产片| 亚洲人在线视频| 国产精品久久久99| 久久久久久久久久久久久9999| 亚洲国产激情| 午夜免费日韩视频| 亚洲国产天堂久久综合| 国产精品久久久久久av福利软件 | 99这里只有精品| 久久久国产精品一区二区三区| 亚洲电影激情视频网站| 欧美网站大全在线观看| 久久中文字幕导航| 这里只有精品在线播放| 欧美国产第一页| 亚洲欧美乱综合| 亚洲三级毛片| 国内精品免费午夜毛片| 欧美日韩p片| 狼人天天伊人久久| 午夜欧美不卡精品aaaaa| 欧美激情精品久久久久| 午夜亚洲性色视频| 日韩亚洲精品电影| 伊大人香蕉综合8在线视| 国产精品久久中文| 欧美国产视频一区二区| 久久国产欧美| 中日韩午夜理伦电影免费| 欧美大片第1页| 久久夜色精品一区| 欧美一区二区视频免费观看| 一区二区三区 在线观看视频| 精品91久久久久| 国产欧美日韩视频| 国产精品久久久一区二区三区| 欧美精品在线免费观看| 久久中文久久字幕| 久久久久一区| 久久精品亚洲精品国产欧美kt∨| 亚洲香蕉网站| 亚洲午夜精品久久久久久浪潮| 亚洲国产婷婷香蕉久久久久久| 米奇777超碰欧美日韩亚洲| 久久av一区二区三区漫画| 亚洲欧美日韩精品久久久| 一区二区三区产品免费精品久久75| 亚洲激情综合| 亚洲精品久久久久| 亚洲精品日本| 99国产精品国产精品毛片| 亚洲日本欧美天堂| 亚洲精品一区中文| 亚洲美女在线观看| 99爱精品视频| 亚洲午夜久久久| 亚洲综合大片69999| 亚洲午夜久久久久久久久电影网| 亚洲视频碰碰| 午夜日韩视频| 久久久精品一区| 免费人成网站在线观看欧美高清| 免费永久网站黄欧美| 欧美jizz19性欧美| 亚洲第一精品在线| 亚洲激情社区| 一区二区三区鲁丝不卡| 亚洲午夜免费视频| 欧美一区在线看| 久久久久久久一区| 欧美大秀在线观看| 欧美日韩视频在线观看一区二区三区| 欧美日韩精品一区二区| 国产精品第2页| 国产最新精品精品你懂的| 亚洲福利视频免费观看| 99热免费精品| 欧美一区二区啪啪| 免费成人高清| 亚洲欧洲在线播放| 亚洲欧美日韩精品一区二区| 久久国产精品99久久久久久老狼 | 欧美日韩爆操| 国产伦精品一区二区三区视频黑人| 国产一区二区三区免费不卡| 亚洲国产三级网| 亚洲欧美激情精品一区二区| 久久影视三级福利片| 亚洲激情欧美| 午夜精品久久久久久久99热浪潮| 狂野欧美激情性xxxx| 欧美日韩一区二区三区高清| 国产曰批免费观看久久久| 亚洲高清不卡av| 亚洲欧美日韩另类| 欧美大片一区| 亚洲欧美成人一区二区三区| 免费观看成人www动漫视频| 国产精品久久久久9999高清| 影音先锋久久久| 午夜亚洲激情| 亚洲日韩欧美视频一区| 欧美在线观看一区二区| 欧美激情bt| 激情成人在线视频| 亚洲欧美一区二区在线观看| 欧美波霸影院| 午夜精品福利电影| 欧美日韩免费区域视频在线观看| 黄色成人在线| 性欧美长视频| 亚洲另类春色国产| 噜噜噜91成人网| 国产自产2019最新不卡| 午夜国产不卡在线观看视频| 亚洲人成艺术| 美女精品视频一区| 精久久久久久久久久久| 性欧美1819性猛交| 亚洲最黄网站| 欧美国产日韩精品免费观看| 在线国产欧美| 久久人人看视频| 亚洲欧美日韩国产中文在线| 欧美午夜电影完整版| 夜夜嗨av色一区二区不卡| 欧美国内亚洲| 蜜桃av噜噜一区| 亚洲激情影院| 亚洲成人在线视频网站| 久久久噜噜噜久久人人看| 国产亚洲毛片在线| 久久激情综合网| 午夜久久一区| 国产日韩欧美在线| 久久精品91| 欧美一区午夜精品| 国产主播一区| 久久免费偷拍视频| 久久精品国产久精国产爱| 韩国视频理论视频久久| 久久久久久久波多野高潮日日| 香蕉视频成人在线观看| 国产欧美精品日韩精品| 久久精品系列| 久久婷婷色综合| 亚洲日本无吗高清不卡| 亚洲精品视频免费在线观看| 欧美日韩一区成人|