a亚洲精品_精品国产91乱码一区二区三区_亚洲精品在线免费观看视频_欧美日韩亚洲国产综合_久久久久久久久久久成人_在线区

首頁 > 編程 > C > 正文

VC中CWinThread類以及和createthread API的區別分析

2020-01-26 15:12:04
字體:
來源:轉載
供稿:網友

本文實例講述了VC中CWinThread類以及和createthread API的區別分析,分享給大家供大家參考。具體分析如下:

CWinThread

CObject
 └CCmdTarget
    └CWinThread

CWinThread對象代表在一個應用程序內運行的線程。運行的主線程通常由CWinApp的派生類提供;CWinApp由CWinThread派生。另外,CWinThread對象允許一給定的應用程序擁有多個線程。

CWinThread支持兩種線程類型:工作者線程(Worker Thread)和用戶界面線程(UI thread)。工作者線程沒有收發消息的功能(沒有消息隊列):例如,在電子表格應用程序中進行后臺計算的線程。

用戶界面線程具有收發消息的功能,并處理從系統收到的消息。CWinApp及其派生類是用戶界面線程的例子。其它用戶界面線程也可由CWinThread直接派生。

CWinThread類的對象存在于線程的生存期。如果你希望改變這個特性,將m_bAutoDelete設為FALSE。

要使你的代碼和MFC是完全線程安全的,CWinThread類是完全必要的。框架使用的用來維護與線程相關的信息的線程局部數據由CWinThread對象管理。由于依賴CWinThread來處理線程局部數據(Thread Local Storage),任何使用MFC的線程必須由MFC創建。例如,由運行時函數_beginthreadex創建的線程不能使用任何MFC API。

為了創建一個線程,調用AfxBeginThread函數。根據你需要工作者線程還是用戶界面線程,有兩種調用AfxBeginThread的格式。如果你需要用戶界面線程,則將指向你的CWinThread派生類的CRuntimeClass的指針傳遞給AfxBeginThread。如果你需要創建工作者線程,則將指向控制函數的指針和控制函數的參數傳遞給AfxBeginThread。對于工作者線程和用戶界面線程,你可以指定可選的參數來修改優先級,堆棧大小,創建標志和安全屬性。

AfxBeginThread線程將返回指向新的CWinThread對象的指針。
與調用AfxBeginThread相反,你可以構造一個CWinThread派生類的對象,然后調用CreateThread。如果你需要在連續創建和終止線程的執行之間重復使用CWinThread對象,這種兩步構造方法非常有用。

CWinThread類成員

數據成員

m_bAutoDelete                      指定線程結束時是否要銷毀對象 
m_hThread                             當前線程的句柄 
m_nThreadID                         當前線程的ID 
m_pMainWnd                         保存指向應用程序的主窗口的指針 
m_pActiveWnd                       指向容器應用程序的主窗口,當一個OLE服務器被現場激活時 

構造函數

CWinThread                           構造一個CWinThread對象 
CreateThread                          開始一個CWinThread對象的執行 

操作

GetMainWnd                          查詢指向線程主窗口的指針 
GetThreadPriority                    獲取當前線程的優先級 
PostThreadMessage               向另外的CWinThread對象傳遞一條消息 
ResumeThread                       減少一個線程的掛起計數 
SetThreadPriority                   設置當前線程的優先級 
SuspendThread                     增加一個線程的掛起計數 

可重載函數

ExitInstance                         重載以進行線程終止時的清理工作 
InitInstance                           重載以實現線程實例的初始化 
OnIdle                                   重載以進行線程特定的空閑操作 
PreTranslateMessage           在消息被發送到Windows函數TranslateMessage和DispatchMessage之前過濾消息 
IsIdleMessage                      檢測特定的消息 
ProcessWndProcException    截獲線程消息和命令處理函數出現的所有未處理的異常 
ProcessMessageFilter           在特定的消息到達應用程序之前截獲消息 
Run                                       線程的具有消息收發功能的控制函數,可重載以定制缺省的消息循環 

AfxBeginThread和CreateThread具體區別

具體說來,CreateThread這個 函數是windows提供給用戶的 API函數,是SDK的標準形式.
AfxBeginThread,是編譯器對原來的CreateThread函數的封裝,用與MFC.
而_beginthread是C的運行庫函數。               
在使用AfxBeginThread時,線程函數的定義為:UINT  _yourThreadFun(LPVOID   pParam)
在使用CreateThread時,線程的函數定義為:  DWORD  WINAPI  _yourThreadFun(LPVOID pParameter)。

兩個的實質都是一樣的,不過AfxBeginThread返回一個CWinThread的指針,就是說他會new一個CWinThread對象,而且這個對象是自動刪除的(在線程運行結束時),給我們帶來的不便就是無法獲得它的狀態,因為隨時都有可能這個指針指向的是一個已經無效的內存區域,所以使用時(如果需要了解它的運行狀況的話)首先CREATE_SUSPENDED讓他掛起,然后m_bAutoDelete=FALSE,接著才ResumeThread,最后不要了delete那個指針。

CreatThread就方便多了,它返回的是一個句柄,如果你不使用CloseHandle的話就可以通過他安全的了解線程狀態,最后不要的時候CloseHandle,Windows才會釋放資源(線程內核對象).
下面我們就來看一下AfxBeginThread函數的內部實現:

復制代碼 代碼如下:
//啟動worker線程
CWinThread* AFXAPI AfxBeginThread(AFX_THREADPROC pfnThreadProc, LPVOID pParam,
    int nPriority, UINT nStackSize, DWORD dwCreateFlags,
    LPSECURITY_ATTRIBUTES lpSecurityAttrs)
{

         ASSERT(pfnThreadProc != NULL);

         CWinThread* pThread = DEBUG_NEW CWinThread(pfnThreadProc, pParam);
         ASSERT_VALID(pThread);

         if (!pThread->CreateThread(dwCreateFlags|CREATE_SUSPENDED, nStackSize,
                  lpSecurityAttrs))
         {
                  pThread->Delete();
                  return NULL;
         }
         VERIFY(pThread->SetThreadPriority(nPriority));
         if (!(dwCreateFlags & CREATE_SUSPENDED))
                  VERIFY(pThread->ResumeThread() != (DWORD)-1);
         return pThread;
}

//啟動UI線程
CWinThread* AFXAPI AfxBeginThread(CRuntimeClass* pThreadClass,int nPriority, UINT nStackSize, DWORD dwCreateFlags, LPSECURITY_ATTRIBUTES lpSecurityAttrs)
{

        ASSERT(pThreadClass != NULL);
        ASSERT(pThreadClass->IsDerivedFrom(RUNTIME_CLASS(CWinThread)));
        CWinThread* pThread = (CWinThread*)pThreadClass->CreateObject();
        if (pThread == NULL)
                AfxThrowMemoryException();
        ASSERT_VALID(pThread);
        pThread->m_pThreadParams = NULL;
        if (!pThread->CreateThread(dwCreateFlags|CREATE_SUSPENDED, nStackSize,
                lpSecurityAttrs))
        {
                pThread->Delete();
                return NULL;
        }
        VERIFY(pThread->SetThreadPriority(nPriority));
        if (!(dwCreateFlags & CREATE_SUSPENDED))
                VERIFY(pThread->ResumeThread() != (DWORD)-1);
        return pThread;
}

主要創建函數是

復制代碼 代碼如下:
pThread->CreateThread(dwCreateFlags|CREATE_SUSPENDED, nStackSize,lpSecurityAttrs))

也就是

復制代碼 代碼如下:
CWinThread::CreateThread

希望本文所述對大家的VC程序設計有所幫助。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

主站蜘蛛池模板: 亚洲一区二区三区观看 | 欧美一区二区三区在线观看 | 欧美a区 | 欧美日韩一二三区 | 国产精品99久久久久久大便 | 国产激情 | 欧美精品网站 | 成人亚洲精品久久久久软件 | 国产精品久久久久国产a级 91国内外精品自在线播放 | 极品美女国产精品免费一区 | 91社影院在线观看 | www.99久| 国产精品日韩 | 亚洲女同老女人女同志 | 国产成人精品一区二区三区视频 | 日韩一区二区三区免费 | 国产全黄 | 中国一级大黄大黄大色毛片 | 国产精品乱码一区二区三区 | 欧美黄色大片网站 | 欧美激情在线狂野欧美精品 | 中文字幕亚洲一区 | 精品久久中文字幕 | 欧美色视 | 最新高清无码专区 | 中文字幕欧美在线观看 | 午夜寂寞影视 | 国产视频一二三区 | 99久久久国产精品 | 亚洲 自拍 另类 欧美 丝袜 | www.4虎| 日韩乱码中文字幕 | 久久亚洲免费 | 日韩欧美国产一区二区 | 国产视频第一页 | 色精品视频 | 久久一级| 国产一区二区三区视频在线观看 | 一区二区在线免费观看 | 国产精品粉嫩白浆在线观看 | 五月综合久久 |