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

franksunny的個人技術空間
獲得人生中的成功需要的專注與堅持不懈多過天才與機會。 ——C.W. Wendte

雖然自己用多線程編程用過一陣子,但是未曾仔細了解過概念,用的也是亂亂的,今天看到一篇線程總結的文章,感覺講的很好,Windows下的多線程也就是了解了線程的概念然后加一同步代碼就行了。

Windows平臺下的多線程編程

?

線程是進程的一條執行路徑,它包含獨立的堆棧和 CPU 寄存器狀態,每個線程共享所有的進程資源,包括打開的文件、信號標識及動態分配的內存等。一個進程內的所有線程使用同一個地址空間,而這些線程的執行由系統調度程序控制,調度程序決定哪個線程可執行以及什么時候執行線程。線程有優先級別,優先權較低的線程必須等到優先權較高的線程執行完后再執行。在多處理器的機器上,調度程序可將多個線程放到不同的處理器上去運行,這樣可使處理器任務平衡,并提高系統的運行效率。

Windows 是一種多任務的操作系統,在 Windows 的一個進程內包含一個或多個線程。 32 Windows 環境下的 Win32 API 提供了多線程應用程序開發所需要的接口函數,而利用VC中提供的標準C庫也可以開發多線程應用程序,相應的MFC類庫封裝了多線程編程的類,用戶在開發時可根據應用程序的需要和特點選擇相應的工具。為了使大家能全面地了解 Windows 多線程編程技術,本文將重點介紹 Win32 API MFC 兩種方式下如何編制多線程程序。

多線程編程在 Win32 方式下和 MFC 類庫支持下的原理是一致的,進程的主線程在任何需要的時候都可以創建新的線程。當線程執行完后,自動終止線程 ; 當進程結束后,所有的線程都終止。所有活動的線程共享進程的資源,因此,在編程時需要考慮在多個線程訪問同一資源時產生沖突的問題。當一個線程正在訪問某進程對象,而另一個線程要改變該對象,就可能會產生錯誤的結果,編程時要解決這個沖突。

?

Win32 API 下的多線程編程

Win32 API Windows 操作系統內核與應用程序之間的界面,它將內核提供的功能進行函數包裝,應用程序通過調用相關函數而獲得相應的系統功能。為了向應用程序提供多線程功能, Win32 API 函數集中提供了一些處理多線程程序的函數集。直接用 Win32 API 進行程序設計具有很多優點 : 基于 Win32 的應用程序執行代碼小,運行效率高,但是它要求程序員編寫的代碼較多,且需要管理所有系統提供給程序的資源。用 Win32 API 直接編寫程序要求程序員對 Windows 系統內核有一定的了解,會占用程序員很多時間對系統資源進行管理,因而程序員的工作效率降低。

?

1. Win32 函數創建和終止線程

Win32 函數庫中提供了操作多線程的函數,包括創建線程、終止線程、建立互斥區等。在應用程序的主線程或者其他活動線程中創建新的線程的函數如下:

HANDLE CreateThread( LPSECURITY_ATTRIBUTES lpThreadAttributes,DWORD dwStackSize,LPTHREAD_START_ROUTINE lpStartAddress,LPVOID lpParameter,DWORD dwCreationFlags,LPDWORD lpThreadId);

如果創建成功則返回線程的句柄,否則返回 NULL 。創建了新的線程后,該線程就開始啟動執行了。但如果在 dwCreationFlags 中使用了 CREATE_SUSPENDED 特性,那么線程并不馬上執行,而是先掛起,等到調用 ResumeThread 后才開始啟動線程,在這個過程中可以調用下面這個函數來設置線程的優先權:

BOOL SetThreadPriority(HANDLE hThread,int nPriority);

當調用線程的函數返回后,線程自動終止。如果需要在線程的執行過程中終止則可調用函數:

VOID ExitThread(DWORD dwExitCode);

如果在線程的外面終止線程,則可調用下面的函數:

BOOL TerminateThread(HANDLE hThread,DWORD dwExitCode);

但應注意 : 該函數可能會引起系統不穩定,而且線程所占用的資源也不釋放。因此,一般情況下,建議不要使用該函數。

如果要終止的線程是進程內的最后一個線程,則線程被終止后相應的進程也應終止。

?

2. 線程的同步

在線程體內,如果該線程完全獨立,與其他線程沒有數據存取等資源操作上的沖突,則可按照通常單線程的方法進行編程。但是,在多線程處理時情況常常不是這樣,線程之間經常要同時訪問一些資源。由于對共享資源進行訪問引起沖突是不可避免的,為了解決這種線程同步問題, Win32 API 提供了多種同步控制對象來幫助程序員解決共享資源訪問沖突。在介紹這些同步對象之前先介紹一下等待函數,因為所有控制對象的訪問控制都要用到這個函數。

Win32 API 提供了一組能使線程阻塞其自身執行的等待函數。這些函數在其參數中的一個或多個同步對象產生了信號,或者超過規定的等待時間才會返回。在等待函數未返回時,線程處于等待狀態,此時線程只消耗很少的 CPU 時間。使用等待函數既可以保證線程的同步,又可以提高程序的運行效率。最常用的等待函數是:

DWORD WaitForSingleObject(HANDLE hHandle DWORD dwMilliseconds);

而函數 WaitForMultipleObject 可以用來同時監測多個同步對象,該函數的聲明為:

DWORD WaitForMultipleObject(DWORD nCount,CONST HANDLE *lpHandles,BOOL bWaitAll,DWORD dwMilliseconds);

1 )互斥體對象

Mutex 對象的狀態在它不被任何線程擁有時才有信號,而當它被擁有時則無信號。 Mutex 對象很適合用來協調多個線程對共享資源的互斥訪問。可按下列步驟使用該對象:

首先,建立互斥體對象,得到句柄:

HANDLE CreateMutex();

然后,在線程可能產生沖突的區域前(即訪問共享資源之前)調用 WaitForSingleObject ,將句柄傳給函數,請求占用互斥對象:

dwWaitResult = WaitForSingleObject(hMutex,5000L);

共享資源訪問結束,釋放對互斥體對象的占用:

ReleaseMutex(hMutex);

互斥體對象在同一時刻只能被一個線程占用,當互斥體對象被一個線程占用時,若有另一線程想占用它,則必須等到前一線程釋放后才能成功。

?

2 )信號對象

信號對象允許同時對多個線程共享資源進行訪問,在創建對象時指定最大可同時訪問的線程數。當一個線程申請訪問成功后,信號對象中的計數器減一,調用 ReleaseSemaphore 函數后,信號對象中的計數器加一。其中,計數器值大于或等于0,但小于或等于創建時指定的最大值。如果一個應用在創建一個信號對象時,將其計數器的初始值設為0,就阻塞了其他線程,保護了資源。等初始化完成后,調用 ReleaseSemaphore 函數將其計數器增加至最大值,則可進行正常的存取訪問。可按下列步驟使用該對象:

首先,創建信號對象:

HANDLE CreateSemaphore();

或者打開一個信號對象:

HANDLE OpenSemaphore();

然后,在線程訪問共享資源之前調用 WaitForSingleObject

共享資源訪問完成后,應釋放對信號對象的占用:

ReleaseSemaphore();

3 )事件對象

事件對象 (Event) 是最簡單的同步對象,它包括有信號和無信號兩種狀態。在線程訪問某一資源之前,需要等待某一事件的發生,這時用事件對象最合適。例如:只有在通信端口緩沖區收到數據后,監視線程才被激活。

事件對象是用 CreateEvent 函數建立的。該函數可以指定事件對象的類和事件的初始狀態。如果是手工重置事件,那么它總是保持有信號狀態,直到用 ResetEvent 函數重置成無信號的事件。如果是自動重置事件,那么它的狀態在單個等待線程釋放后會自動變為無信號的。用 SetEvent 可以把事件對象設置成有信號狀態。在建立事件時,可以為對象命名,這樣其他進程中的線程可以用 OpenEvent 函數打開指定名字的事件對象句柄。

4 )排斥區對象

在排斥區中異步執行時,它只能在同一進程的線程之間共享資源處理。雖然此時上面介紹的幾種方法均可使用,但是,使用排斥區的方法則使同步管理的效率更高。

使用時先定義一個 CR99vICAL_SECTION 結構的排斥區對象,在進程使用之前調用如下函數對對象進行初始化 :

VOID InitializeCriticalSection( LPCRITICAL_SECTION );

當一個線程使用排斥區時,調用函數: EnterCriticalSection 或者 TryEnterCriticalSection;

當要求占用、退出排斥區時,調用函數 LeaveCriticalSection ,釋放對排斥區對象的占用,供其他線程使用。

?

基于MFC的多線程編程

?

MFC 是微軟的 VC 開發集成環境中提供給程序員的基礎函數庫,它用類庫的方式將 Win32 API 進行封裝,以類的方式提供給開發者。由于其快速、簡捷、功能強大等特點深受廣大開發者喜愛。因此,建議使用 MFC 類庫進行應用程序的開發。

VC++ 附帶的 MFC 類庫中,提供了對多線程編程的支持,基本原理與基于 Win32 API 的設計一致,但由于 MFC 對同步對象做了封裝,因此實現起來更加方便,避免了對象句柄管理上的煩瑣工作。

MFC 中,線程分為兩種:工作線程和用戶接口線程。工作線程與前面所述的線程一致,用戶接口線程是一種能夠接收用戶的輸入、處理事件和消息的線程。

1. 工作線程

工作線程編程較為簡單,設計思路與前面所講的基本一致 : 一個基本函數代表了一個線程,創建并啟動線程后,線程進入運行狀態 ; 如果線程用到共享資源,則需要進行資源同步處理。這種方式創建線程并啟動線程時可調用函數:

CWinThread*AfxBeginThread( AFX_THREADPROC pfnThreadProc, LPVOID pParam,int nPriority= THREAD_PRIOR ITY _NORMAL,UINT nStackSize =0,DWORD dwCreateFlags=0, LPSECURITY_ATTRIBUTES lpSecurityAttrs = NULL);

參數 pfnThreadProc 是線程執行體函數,函數原形為 : UINT ThreadFunction( LPVOID pParam)

參數 pParam 是傳遞給執行函數的參數;

參數 nPriority 是線程執行權限,可選值:

THREAD_PRIOR ITY _NORMAL THREAD_PRIOR ITY _LOWEST THREAD_PRIOR ITY _HIGHEST THREAD_PRIOR ITY _IDLE

參數 dwCreateFlags 是線程創建時的標志,可取值 CREATE_SUSPENDED ,表示線程創建后處于掛起狀態,調用 ResumeThread 函數后線程繼續運行,或者取值 0” 表示線程創建后處于運行狀態。

返回值是 CWinThread 類對象指針,它的成員變量 m_hThread 為線程句柄,在 Win32 API 方式下對線程操作的函數參數都要求提供線程的句柄,所以當線程創建后可以使用所有 Win32 API 函數對 pWinThread->m_Thread 線程進行相關操作。

?

注意:如果在一個類對象中創建和啟動線程時,應將線程函數定義成類外的全局函數。 (這句話是否必要,我也不清楚)

?

2. 用戶接口線程

基于 MFC 的應用程序有一個應用對象,它是 CWinApp 派生類的對象,該對象代表了應用進程的主線程。當線程執行完并退出線程時,由于進程中沒有其他線程存在,進程自動結束。類 C inApp C inThread 派生出來, C inThread 是用戶接口線程的基本類。我們在編寫用戶接口線程時,需要從 C inThread 派生我們自己的線程類, ClassWizard 可以幫助我們完成這個工作。

先用 ClassWizard 派生一個新的類,設置基類為 CwinThread 。注意:類的 DECLARE_DYNCREATE IMPLEMENT_DYNCREATE 宏是必需的,因為創建線程時需要動態創建類的對象。根據需要可將初始化和結束代碼分別放在類的 InitInstance ExitInstance 函數中。如果需要創建窗口,則可在 InitInstance 函數中完成。然后創建線程并啟動線程。可以用兩種方法來創建用戶接口線程, MFC 提供了兩個版本的 AfxBeginThread 函數,其中一個用于創建用戶接口線程。第二種方法分為兩步進行:首先,調用線程類的構造函數創建一個線程對象;其次,調用 CWinThread::CreateThread 函數來創建該線程。線程建立并啟動后,在線程函數執行過程中一直有效。如果是線程對象,則在對象刪除之前,先結束線程。 CWinThread 已經為我們完成了線程結束的工作。

?

3. 線程同步

前面我們介紹了 Win32 API 提供的幾種有關線程同步的對象,在 MFC 類庫中對這幾個對象進行了類封裝,它們有一個共同的基類 CSyncObject ,它們的對應關系為 : Semaphore 對應 CSemaphore Mutex 對應 CMutex Event 對應 CEvent CriticalSection 對應 CCriticalSection 。另外, MFC 對兩個等待函數也進行了封裝,即 CSingleLock CMultiLock 。因四個對象用法相似,在這里就以 CMutex 為例進行說明:

創建一個 CMutex 對象 :

CMutex mutex(FALSE,NULL,NULL); CMutex mutex;

當各線程要訪問共享資源時使用下面代碼:

CSingleLock sl(&mutex);

sl.Lock();

if(sl.IsLocked())

// 對共享資源進行操作 ...

sl.Unlock();

?

結束語

如果用戶的應用程序需要多個任務同時進行相應的處理,則使用多線程是較理想的選擇。這里,提醒大家注意的是在多線程編程時要特別小心處理資源共享問題以及多線程調試問題。筆者準備了幾個實例,如大家需要的話,可以和筆者聯系。

posted on 2007-01-26 22:10 frank.sunny 閱讀(2553) 評論(0)  編輯 收藏 引用 所屬分類: MFC相關技術

常用鏈接

留言簿(13)

隨筆分類

個人其它博客

基礎知識鏈接

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美日本高清| 亚洲精品美女免费| 激情另类综合| 韩日欧美一区| 在线视频观看日韩| 亚洲国产欧美不卡在线观看| 亚洲国产精品黑人久久久| 亚洲国产精品国自产拍av秋霞| 在线精品福利| 亚洲国产精品一区二区三区| 亚洲精品欧美在线| 亚洲欧美激情视频| 久久久精品国产一区二区三区| 久久只精品国产| 亚洲人成在线观看一区二区| 99av国产精品欲麻豆| 亚洲欧美久久久| 久热精品视频在线观看一区| 欧美精品一区二| 国产精一区二区三区| 又紧又大又爽精品一区二区| 99在线热播精品免费| 欧美在线短视频| 欧美高清在线播放| 一区二区三区免费网站| 久久不射中文字幕| 欧美视频一区在线| 在线播放中文一区| 欧美一级久久久久久久大片| 欧美成人精品在线| 亚洲一区二区三区精品在线| 牛牛国产精品| 国产视频一区二区三区在线观看| 亚洲欧洲视频在线| 久久久久.com| 亚洲一区欧美一区| 欧美日韩国产色站一区二区三区| 狠狠色丁香久久婷婷综合丁香| 亚洲一区999| 亚洲人永久免费| 久久午夜影视| 国产一区二区久久| 亚洲欧美在线另类| 亚洲精品影视| 欧美国产日韩亚洲一区| 激情综合激情| 久久久女女女女999久久| 一区二区三区三区在线| 欧美国产日韩视频| 亚洲精品乱码久久久久久黑人| 玖玖综合伊人| 久久精品国产99国产精品澳门| 欧美午夜精品| 久久亚洲私人国产精品va| 精品成人在线| 欧美伊人久久| 一区二区三区高清| 欧美日韩色综合| 99精品黄色片免费大全| 亚洲国产精品成人va在线观看| 久久精品在线免费观看| 黄色影院成人| 久久永久免费| 久久久亚洲精品一区二区三区| 国产亚洲免费的视频看| 久久久人成影片一区二区三区| 性色av一区二区怡红| 国产一区 二区 三区一级| 久久久福利视频| 欧美中文字幕| 亚洲国产第一页| 亚洲第一级黄色片| 欧美日韩高清在线播放| 亚洲一区亚洲| 香蕉久久夜色精品国产| 国产在线精品二区| 免费不卡中文字幕视频| 欧美14一18处毛片| 亚洲专区一区二区三区| 亚洲欧美日韩国产中文| 伊人蜜桃色噜噜激情综合| 欧美激情视频给我| 欧美日韩三区| 久久久久九九九九| 欧美精品久久久久久久久久| 性色一区二区三区| 老司机久久99久久精品播放免费| 亚洲欧洲在线一区| 亚洲天堂男人| 激情综合色综合久久| 亚洲欧洲另类| 国产精品一区二区在线观看| 久久天堂成人| 欧美午夜女人视频在线| 久久综合九色九九| 欧美日韩一区二区免费在线观看 | 国产精品久久久久久久浪潮网站| 午夜免费日韩视频| 久久影院午夜片一区| 一区二区三区高清视频在线观看| 亚洲欧美日韩一区二区在线 | 久久久噜噜噜久久| 欧美日韩国产免费| 久久在线观看视频| 国产精品乱子久久久久| 蜜桃av综合| 欧美日韩午夜剧场| 亚洲乱码视频| 欧美一区国产在线| 亚洲午夜精品国产| 久久国产精彩视频| 在线国产精品播放| 亚洲人成人一区二区在线观看| 国产精品久久久99| 亚洲黄色大片| 一区二区自拍| 香蕉成人伊视频在线观看| 亚洲婷婷在线| 欧美高清在线精品一区| 久久日韩粉嫩一区二区三区| 欧美精品乱码久久久久久按摩| 亚洲美女在线观看| 亚洲影院色在线观看免费| 日韩亚洲成人av在线| 欧美国产欧美亚州国产日韩mv天天看完整| 亚洲欧美在线观看| 欧美精品一二三| 欧美成人激情视频| 精品动漫3d一区二区三区免费版| 亚洲视频在线一区观看| 在线视频精品| 欧美日韩你懂的| 亚洲精品视频在线观看网站| 亚洲精品一二区| 欧美国产精品久久| 免费黄网站欧美| 樱桃国产成人精品视频| 久久av资源网| 久热这里只精品99re8久| 国产在线一区二区三区四区| 欧美一区二区三区男人的天堂| 久久国产福利| 激情成人亚洲| 久久久人人人| 欧美激情女人20p| 日韩视频精品在线观看| 欧美精品乱人伦久久久久久 | 久久久久免费| 韩国三级在线一区| 欧美一区三区三区高中清蜜桃| 欧美一区三区二区在线观看| 国产乱码精品一区二区三区不卡 | 久久精品国产欧美亚洲人人爽 | 一区二区成人精品| 欧美一区二区视频观看视频| 国产欧美日韩视频一区二区三区| 欧美1区2区| 久久精品二区亚洲w码| 狼人社综合社区| 欧美激情中文不卡| 99国产精品99久久久久久| 欧美日韩大片一区二区三区| 99精品欧美一区二区三区| 亚洲一区免费看| 国产亚洲欧洲一区高清在线观看| 欧美中文在线观看| 亚洲国产一区视频| 中日韩视频在线观看| 国产欧美日韩伦理| 欧美1区3d| 亚洲一区二区三区视频播放| 久久综合色影院| 亚洲午夜黄色| 精品成人国产在线观看男人呻吟| 免费在线视频一区| 亚洲综合精品四区| 欧美高清视频一区二区| 亚洲一区二区三区中文字幕| 好吊一区二区三区| 欧美日韩在线观看一区二区| 久久成人国产| 亚洲视频日本| 91久久久久久国产精品| 久久久久久久尹人综合网亚洲| 日韩一二三在线视频播| 狠狠色丁香久久综合频道| 欧美午夜在线观看| 久久综合狠狠综合久久综合88| 一区二区三区精品视频在线观看| 免费观看一级特黄欧美大片| 午夜视频在线观看一区二区| 亚洲精品一区在线观看| 韩国av一区二区| 国产精品日韩| 欧美日韩视频一区二区| 欧美中文在线观看国产| 亚洲青色在线| 国产在线精品一区二区中文 | 小处雏高清一区二区三区| 亚洲高清久久久|