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

   [前言:]當前流行的Windows操作系統,它能同時運行幾個程序(獨立運行的程序又稱之為進程),對于同一個程序,它又可以分成若干個獨立的執行流,我們稱之為線程,線程提供了多任務處理的能力。用進程和線程的觀點來研究軟件是當今普遍采用的方法,進程和線程的概念的出現,對提高軟件的并行性有著重要的意義。現在的應用軟件無一不是多線程多任務處理,單線城的軟件是不可想象的。因此掌握多線程多任務設計方法對每個程序員都是必需要掌握的。本文針對多線程技術在應用中經常遇到的問題,如線程間的通信、同步等,對它們分別進行探討。

   一、 理解線程

   要講解線程,不得不說一下進程,進程是應用程序的執行實例,每個進程是由私有的虛擬地址空間、代碼、數據和其它系統資源組成。進程在運行時創建的資源隨著進程的終止而死亡。線程的基本思想很簡單,它是一個獨立的執行流,是進程內部的一個獨立的執行單元,相當于一個子程序,它對應Visual C++中的CwinThread類的對象。單獨一個執行程序運行時,缺省的運行包含的一個主線程,主線程以函數地址的形式,如main或WinMain函數,提供程序的啟動點,當主線程終止時,進程也隨之終止,但根據需要,應用程序又可以分解成許多獨立執行的線程,每個線程并行的運行在同一進程中。

   一個進程中的所有線程都在該進程的虛擬地址空間中,使用該進程的全局變量和系統資源。操作系統給每個線程分配不同的CPU時間片,在某一個時刻,CPU只執行一個時間片內的線程,多個時間片中的相應線程在CPU內輪流執行,由于每個時間片時間很短,所以對用戶來說,仿佛各個線程在計算機中是并行處理的。操作系統是根據線程的優先級來安排CPU的時間,優先級高的線程優先運行,優先級低的線程則繼續等待。

   線程被分為兩種:用戶界面線程和工作線程(又稱為后臺線程)。用戶界面線程通常用來處理用戶的輸入并響應各種事件和消息,其實,應用程序的主執行線程CWinAPP對象就是一個用戶界面線程,當應用程序啟動時自動創建和啟動,同樣它的終止也意味著該程序的結束,進城終止。工作者線程用來執行程序的后臺處理任務,比如計算、調度、對串口的讀寫操作等,它和用戶界面線程的區別是它不用從CwinThread類派生來創建,對它來說最重要的是如何實現工作線程任務的運行控制函數。工作線程和用戶界面線程啟動時要調用同一個函數的不同版本;最后需要讀者明白的是,一個進程中的所有線程共享它們父進程的變量,但同時每個線程可以擁有自己的變量。
   二、 線程的管理和操作

   1. 線程的啟動

   創建一個用戶界面線程,首先要從類CwinThread產生一個派生類,同時必須使用DECLARE_DYNCREATE和IMPLEMENT_DYNCREATE來聲明和實現這個CwinThread派生類。

   第二步是根據需要重載該派生類的一些成員函數如:ExitInstance();InitInstance();OnIdle();PreTranslateMessage()等函數,最后啟動該用戶界面線程,調用AfxBeginThread()函數的一個版本:CWinThread* AfxBeginThread( CRuntimeClass* pThreadClass, int nPriority = THREAD_PRIORITY_NORMAL, UINT nStackSize = 0, DWORD dwCreateFlags = 0, LPSECURITY_ATTRIBUTES lpSecurityAttrs = NULL );其中第一個參數為指向定義的用戶界面線程類指針變量,第二個參數為線程的優先級,第三個參數為線程所對應的堆棧大小,第四個參數為線程創建時的附加標志,缺省為正常狀態,如為CREATE_SUSPENDED則線程啟動后為掛起狀態。

   對于工作線程來說,啟動一個線程,首先需要編寫一個希望與應用程序的其余部分并行運行的函數如Fun1(),接著定義一個指向CwinThread對象的指針變量*pThread,調用AfxBeginThread(Fun1,param,priority)函數,返回值付給pThread變量的同時一并啟動該線程來執行上面的Fun1()函數,其中Fun1是線程要運行的函數的名字,也既是上面所說的控制函數的名字,param是準備傳送給線程函數Fun1的任意32位值,priority則是定義該線程的優先級別,它是預定義的常數,讀者可參考MSDN。

   2.線程的優先級

   以下的CwinThread類的成員函數用于線程優先級的操作:

int GetThreadPriority();
BOOL SetThradPriority()(int nPriority);

上述的二個函數分別用來獲取和設置線程的優先級,這里的優先級,是相對于該線程所處的優先權層次而言的,處于同一優先權層次的線程,優先級高的線程先運行;處于不同優先權層次上的線程,誰的優先權層次高,誰先運行。至于優先級設置所需的常數,自己參考MSDN就可以了,要注意的是要想設置線程的優先級,這個線程在創建時必須具有THREAD_SET_INFORMATION訪問權限。對于線程的優先權層次的設置,CwinThread類沒有提供相應的函數,但是可以通過Win32 SDK函數GetPriorityClass()和SetPriorityClass()來實現。

   3.線程的懸掛、恢復

   CwinThread類中包含了應用程序懸掛和恢復它所創建的線程的函數,其中SuspendThread()用來懸掛線程,暫停線程的執行;ResumeThread()用來恢復線程的執行。如果你對一個線程連續若干次執行SuspendThread(),則需要連續執行相應次的ResumeThread()來恢復線程的運行。

   4.結束線程

   終止線程有三種途徑,線程可以在自身內部調用AfxEndThread()來終止自身的運行;可以在線程的外部調用BOOL TerminateThread( HANDLE hThread, DWORD dwExitCode )來強行終止一個線程的運行,然后調用CloseHandle()函數釋放線程所占用的堆棧;第三種方法是改變全局變量,使線程的執行函數返回,則該線程終止。下面以第三種方法為例,給出部分代碼:

////////////////////////////////////////////////////////////////
//////CtestView message handlers
/////Set to True to end thread
Bool bend=FALSE;//定義的全局變量,用于控制線程的運行
//The Thread Function
UINT ThreadFunction(LPVOID pParam)//線程函數
{
while(!bend)
{Beep(100,100);
Sleep(1000);
}
return 0;
}
CwinThread *pThread;
HWND hWnd;
/////////////////////////////////////////////////////////////
Void CtestView::OninitialUpdate()
{
hWnd=GetSafeHwnd();
pThread=AfxBeginThread(ThradFunction,hWnd);//啟動線程
pThread->m_bAutoDelete=FALSE;//線程為手動刪除
Cview::OnInitialUpdate();
}
////////////////////////////////////////////////////////////////
Void CtestView::OnDestroy()
{ bend=TRUE;//改變變量,線程結束
WaitForSingleObject(pThread->m_hThread,INFINITE);//等待線程結束
delete pThread;//刪除線程
Cview::OnDestroy();
}
   三、 線程之間的通信

   通常情況下,一個次級線程要為主線程完成某種特定類型的任務,這就隱含著表示在主線程和次級線程之間需要建立一個通信的通道。一般情況下,有下面的幾種方法實現這種通信任務:使用全局變量(上一節的例子其實使用的就是這種方法)、使用事件對象、使用消息。這里我們主要介紹后兩種方法。

   1. 利用用戶定義的消息通信

   在Windows程序設計中,應用程序的每一個線程都擁有自己的消息隊列,甚至工作線程也不例外,這樣一來,就使得線程之間利用消息來傳遞信息就變的非常簡單。首先用戶要定義一個用戶消息,如下所示:#define WM_USERMSG WMUSER+100;在需要的時候,在一個線程中調用

::PostMessage((HWND)param,WM_USERMSG,0,0)

CwinThread::PostThradMessage()

來向另外一個線程發送這個消息,上述函數的四個參數分別是消息將要發送到的目的窗口的句柄、要發送的消息標志符、消息的參數WPARAM和LPARAM。下面的代碼是對上節代碼的修改,修改后的結果是在線程結束時顯示一個對話框,提示線程結束:

UINT ThreadFunction(LPVOID pParam)
{
while(!bend)
{
Beep(100,100);
Sleep(1000);
}
::PostMessage(hWnd,WM_USERMSG,0,0);
return 0;
}
////////WM_USERMSG消息的響應函數為OnThreadended(WPARAM wParam,LPARAM lParam)
LONG CTestView::OnThreadended(WPARAM wParam,LPARAM lParam)
{
AfxMessageBox("Thread ended.");
Retrun 0;
}

上面的例子是工作者線程向用戶界面線程發送消息,對于工作者線程,如果它的設計模式也是消息驅動的,那么調用者可以向它發送初始化、退出、執行某種特定的處理等消息,讓它在后臺完成。在控制函數中可以直接使用::GetMessage()這個SDK函數進行消息分檢和處理,自己實現一個消息循環。GetMessage()函數在判斷該線程的消息隊列為空時,線程將系統分配給它的時間片讓給其它線程,不無效的占用CPU的時間,如果消息隊列不為空,就獲取這個消息,判斷這個消息的內容并進行相應的處理。

   2.用事件對象實現通信

   在線程之間傳遞信號進行通信比較復雜的方法是使用事件對象,用MFC的Cevent類的對象來表示。事件對象處于兩種狀態之一:有信號和無信號,線程可以監視處于有信號狀態的事件,以便在適當的時候執行對事件的操作。上述例子代碼修改如下:

////////////////////////////////////////////////////////////////////
Cevent threadStart,threadEnd;
////////////////////////////////////////////////////////////////////
UINT ThreadFunction(LPVOID pParam)
{
::WaitForSingleObject(threadStart.m_hObject,INFINITE);
AfxMessageBox("Thread start.");
while(!bend)
{
Beep(100,100);
Sleep(1000);
Int result=::WaitforSingleObject(threadEnd.m_hObject,0);
//等待threadEnd事件有信號,無信號時線程在這里懸停
If(result==Wait_OBJECT_0)
Bend=TRUE;
}
::PostMessage(hWnd,WM_USERMSG,0,0);
return 0;
}
/////////////////////////////////////////////////////////////
Void CtestView::OninitialUpdate()
{
hWnd=GetSafeHwnd();
threadStart.SetEvent();//threadStart事件有信號
pThread=AfxBeginThread(ThreadFunction,hWnd);//啟動線程
pThread->m_bAutoDelete=FALSE;
Cview::OnInitialUpdate();
}
////////////////////////////////////////////////////////////////
Void CtestView::OnDestroy()
{ threadEnd.SetEvent();
WaitForSingleObject(pThread->m_hThread,INFINITE);
delete pThread;
Cview::OnDestroy();
}

運行這個程序,當關閉程序時,才顯示提示框,顯示"Thread ended"
   四、 線程之間的同步

   前面我們講過,各個線程可以訪問進程中的公共變量,所以使用多線程的過程中需要注意的問題是如何防止兩個或兩個以上的線程同時訪問同一個數據,以免破壞數據的完整性。保證各個線程可以在一起適當的協調工作稱為線程之間的同步。前面一節介紹的事件對象實際上就是一種同步形式。Visual C++中使用同步類來解決操作系統的并行性而引起的數據不安全的問題,MFC支持的七個多線程的同步類可以分成兩大類:同步對象(CsyncObject、Csemaphore、Cmutex、CcriticalSection和Cevent)和同步訪問對象(CmultiLock和CsingleLock)。本節主要介紹臨界區(critical section)、互斥(mutexe)、信號量(semaphore),這些同步對象使各個線程協調工作,程序運行起來更安全。

   1. 臨界區

   臨界區是保證在某一個時間只有一個線程可以訪問數據的方法。使用它的過程中,需要給各個線程提供一個共享的臨界區對象,無論哪個線程占有臨界區對象,都可以訪問受到保護的數據,這時候其它的線程需要等待,直到該線程釋放臨界區對象為止,臨界區被釋放后,另外的線程可以強占這個臨界區,以便訪問共享的數據。臨界區對應著一個CcriticalSection對象,當線程需要訪問保護數據時,調用臨界區對象的Lock()成員函數;當對保護數據的操作完成之后,調用臨界區對象的Unlock()成員函數釋放對臨界區對象的擁有權,以使另一個線程可以奪取臨界區對象并訪問受保護的數據。同時啟動兩個線程,它們對應的函數分別為WriteThread()和ReadThread(),用以對公共數組組array[]操作,下面的代碼說明了如何使用臨界區對象:

#include "afxmt.h"
int array[10],destarray[10];
CCriticalSection Section;
////////////////////////////////////////////////////////////////////////
UINT WriteThread(LPVOID param)
{Section.Lock();
for(int x=0;x<10;x++)
array[x]=x;
Section.Unlock();
}
UINT ReadThread(LPVOID param)
{
Section.Lock();
For(int x=0;x<10;x++)
Destarray[x]=array[x];
Section.Unlock();
}

上述代碼運行的結果應該是Destarray數組中的元素分別為1-9,而不是雜亂無章的數,如果不使用同步,則不是這個結果,有興趣的讀者可以實驗一下。
   2. 互斥

   互斥與臨界區很相似,但是使用時相對復雜一些,它不僅可以在同一應用程序的線程間實現同步,還可以在不同的進程間實現同步,從而實現資源的安全共享。互斥與Cmutex類的對象相對應,使用互斥對象時,必須創建一個CSingleLock或CMultiLock對象,用于實際的訪問控制,因為這里的例子只處理單個互斥,所以我們可以使用CSingleLock對象,該對象的Lock()函數用于占有互斥,Unlock()用于釋放互斥。實現代碼如下:

#include "afxmt.h"
int array[10],destarray[10];
CMutex Section;

/////////////////////////////////////////////////////////////
UINT WriteThread(LPVOID param)
{ CsingleLock singlelock;
singlelock (&Section);
singlelock.Lock();
for(int x=0;x<10;x++)
array[x]=x;
singlelock.Unlock();
}
UINT ReadThread(LPVOID param)
{ CsingleLock singlelock;
singlelock (&Section);
singlelock.Lock();

For(int x=0;x<10;x++)
Destarray[x]=array[x];
singlelock.Unlock();

}

   3. 信號量

   信號量的用法和互斥的用法很相似,不同的是它可以同一時刻允許多個線程訪問同一個資源,創建一個信號量需要用Csemaphore類聲明一個對象,一旦創建了一個信號量對象,就可以用它來對資源的訪問技術。要實現計數處理,先創建一個CsingleLock或CmltiLock對象,然后用該對象的Lock()函數減少這個信號量的計數值,Unlock()反之。下面的代碼分別啟動三個線程,執行時同時顯示二個消息框,然后10秒后第三個消息框才得以顯示。

/////////////////////////////////////////////////////////////////
Csemaphore *semaphore;
Semaphore=new Csemaphore(2,2);
HWND hWnd=GetSafeHwnd();
AfxBeginThread(threadProc1,hWnd);
AfxBeginThread(threadProc2,hWnd);
AfxBeginThread(threadProc3,hWnd);
//////////////////////////////////////////////////////////////////////
UINT ThreadProc1(LPVOID param)
{CsingleLock singelLock(semaphore);
singleLock.Lock();
Sleep(10000);
::MessageBox((HWND)param,"Thread1 had access","Thread1",MB_OK);
return 0;
}
UINT ThreadProc2(LPVOID param)
{CSingleLock singelLock(semaphore);
singleLock.Lock();
Sleep(10000);
::MessageBox((HWND)param,"Thread2 had access","Thread2",MB_OK);
return 0;
}
UINT ThreadProc3(LPVOID param)
{CsingleLock singelLock(semaphore);
singleLock.Lock();
Sleep(10000);
::MessageBox((HWND)param,"Thread3 had access","Thread3",MB_OK);
return 0;
}


   對復雜的應用程序來說,線程的應用給應用程序提供了高效、快速、安全的數據處理能力。本文講述了線程中經常遇到的問題,希望對讀者朋友有一定的幫助。
Posted on 2005-11-08 19:13 艾凡赫 閱讀(1196) 評論(0)  編輯 收藏 引用 所屬分類: 多線程
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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一区| 欧美日韩午夜剧场| 久久精品一本久久99精品| 欧美高清视频在线观看| 亚洲欧美日韩国产成人精品影院| 国产一区观看| 国产一区自拍视频| 亚洲乱码精品一二三四区日韩在线 | 欧美亚洲在线播放| 欧美激情亚洲激情| 亚洲激情在线| 在线国产亚洲欧美| 国产日韩欧美综合| 国产精品成人久久久久| 99精品视频免费观看视频| 国产视频久久久久| 国产伦精品一区二区三区照片91| 欧美日韩八区| 国产精品欧美经典| 国产伦精品一区二区三区在线观看 | 国语自产偷拍精品视频偷 | 亚洲视频每日更新| av成人激情| 国产精品日产欧美久久久久| 欧美日本簧片| 国产毛片精品视频| 亚洲激情在线| 亚洲欧美卡通另类91av| 久久精品国产一区二区三| 久久久久久久激情视频| 六月婷婷久久| 日韩亚洲精品电影| 欧美专区中文字幕| 欧美精品在欧美一区二区少妇| 欧美日韩国产电影| 一区福利视频| 性做久久久久久久免费看| 日韩视频在线一区| 久久久91精品国产一区二区三区| 亚洲二区在线| 亚洲综合日韩在线| 欧美96在线丨欧| 激情校园亚洲| 欧美午夜激情视频| 韩国一区二区在线观看| 一区免费观看视频| 亚洲第一黄色网| 99在线精品免费视频九九视| 欧美91大片| 99精品国产福利在线观看免费| 亚洲国产精品999| 久久亚洲一区二区三区四区| 亚洲午夜一区二区三区| 欧美国产高潮xxxx1819| 久久天天躁狠狠躁夜夜av| 国产精品日本精品| 欧美一区二区在线免费播放| 一区二区日韩免费看| 欧美日韩一区二区三区| 亚洲欧美日本伦理| 亚洲欧美日韩国产另类专区| 国产精品毛片在线看| 国产一区深夜福利| 久久免费一区| 欧美一区二区日韩| 一区二区三区在线免费视频| 欧美成人精品高清在线播放| 欧美极品aⅴ影院| 欧美一区2区视频在线观看| 亚洲免费视频成人| 亚洲精品之草原avav久久| 日韩亚洲成人av在线| 激情丁香综合| 久久在线91| 国产精品伦理| 亚洲精品日本| 欧美片在线播放| 久久一区精品| 久久激情一区| 欧美激情日韩| 老色批av在线精品| 国产精品午夜久久| 亚洲国产精品一区二区久| 国产欧美一区二区白浆黑人| 亚洲欧洲一区二区三区久久| 国产日本亚洲高清| 久久精品99久久香蕉国产色戒| 中文在线资源观看网站视频免费不卡 | 久久国产精品久久久久久久久久 | 亚洲午夜久久久| 一本色道久久88精品综合| 亚洲一区欧美| 亚洲国产高清在线| 久热精品视频| 国产精品久久久久久久久久久久久久| 亚洲国产日韩在线一区模特| 免费成人av在线看| 久久婷婷国产综合精品青草| 国内一区二区三区在线视频| 91久久夜色精品国产网站| 麻豆91精品91久久久的内涵| 亚洲图片激情小说| 亚洲女性裸体视频| 99精品欧美一区二区三区| 亚洲美女在线视频| 亚洲国产另类精品专区| 国产精品五月天| 久久中文字幕导航| 国产精品爽黄69| 嫩模写真一区二区三区三州| 国产毛片一区二区| 亚洲国产欧美不卡在线观看| 国产精品综合不卡av| 在线亚洲美日韩| 一本久久综合亚洲鲁鲁| 久久久久久亚洲综合影院红桃| 亚洲精品国产日韩| 久久精品99国产精品| 99国产精品久久久久久久久久| 久久国产一区| 亚洲欧美日韩第一区| 欧美激情成人在线视频| 亚洲一区一卡| 在线观看国产成人av片| 欧美激情综合五月色丁香小说| 中日韩在线视频| 亚洲黄色在线看| 久久精品国产清自在天天线| 亚洲影院在线观看| 亚洲女同精品视频| 亚洲人在线视频| 永久域名在线精品| 国语精品中文字幕| 国产免费观看久久| 国产精品九九| 国产精品免费福利| 欧美香蕉视频| 国产免费成人av| 国产伦精品一区| 国产欧美精品xxxx另类| 欧美精品一二三| 欧美日韩免费一区| 欧美日本中文| 91久久精品久久国产性色也91| 一区二区欧美精品| 亚洲专区一二三| 亚洲欧美日韩一区二区三区在线| 一区二区三区精品| 欧美一级大片在线观看| 久久亚洲精品网站| 亚洲激情国产| 一本色道久久综合一区| 一区二区三区中文在线观看| 激情av一区| 激情视频亚洲| 一区二区精品在线| 亚洲天堂网在线观看| 欧美在线视屏| 欧美午夜不卡影院在线观看完整版免费| 国产精品sss| 亚洲人成久久| 久久久久一区| 亚洲视频免费| 久久久久久久网| 国产一级久久| 女人天堂亚洲aⅴ在线观看| 国产精品区二区三区日本| 亚洲免费电影在线| 久久一区二区三区四区| 在线视频你懂得一区二区三区| 久久久久久自在自线| 国产精品九九| 亚洲欧美另类在线观看| 欧美成年网站| 欧美在线首页| 国模精品娜娜一二三区| 羞羞色国产精品| 亚洲欧美日韩一区在线| 久久精品在线播放| 国产一区二区三区高清| 亚洲免费在线观看视频| 亚洲在线不卡| 欧美成人一区二区三区在线观看| 欧美一区二区黄| 在线观看亚洲精品视频| 亚洲二区在线观看| 亚洲一二三区在线| 国产精品久久一区二区三区| 蜜臀99久久精品久久久久久软件| 美女视频网站黄色亚洲| 欧美一级视频| 欧美午夜电影一区| 欧美激情在线播放| 国模套图日韩精品一区二区| 夜夜精品视频一区二区| 欧美日韩精品高清| 91久久久在线|