MFC啟動和關閉線程


1、 啟動線程:

CWinThread* AfxBeginThread( 線程函數,this );

2、通常導致線程終止的兩種情況是:控制函數退出或不允許線程完成運行。如果字處理器使用后台打印線程,若成功完成打印,則控制函數將正常終止。但是,如果用戶要取消打印,后台打印線程則不得不提前終止。本主題介紹如何實現每一種情況,以及在終止后如何獲取線程的退出代碼。

(1)正常線程終止

對於輔助線程,正常線程終止很簡單:退出控制函數並返回表示終止原因的值。可以使用 函數或 return 語句。一般情況下,0 表示成功完成,但這取決於您自己。

對於用戶界面線程,該過程也很簡單:從用戶界面線程內調用 Platform SDK 中的 。PostQuitMessage 采用的唯一參數是線程的退出代碼。對於輔助線程,0 通常表示成功完成。

(2)過早的線程終止

過早終止線程幾乎一樣簡單:從線程內調用 。將所需的退出代碼作為唯一參數傳遞。這將停止執行線程、解除對線程堆棧的分配、分離附加到線程的所有 DLL 並從內存中刪除線程對象。

必須從要終止的線程內調用 AfxEndThread。如果要從其他線程終止線程,必須設置兩個線程間的通信方法。

舉一個例子:

可以創建一個信號量,用WaitForSingleObject函數來檢測該信號量的狀態。

成員變量 m_hThreadEvent;

m_hThreadEvent = CreateEvent( NULL, FALSE, FALSE, NULL );

線程的執行函數:

for( ; ;)
{        
    DWORD dwRetVal;
    dwRetVal = WaitForSingleObject( m_hThreadEvent, 100 );
    if ( dwRetVal == WAIT_TIMEOUT )
    {
        // TODO:
    }
    else
    {
        // stop receive text thread.
        DWORD dwExitCode;
        GetExitCodeThread( m_pThreadRecv->m_hThread, &dwExitCode );
        AfxEndThread( dwExitCode, TRUE );
    }
}

要結束線程時,使用SetEvent,將信號量置為有信號。

該線程是在信號量有信號時,退出。

(3)TerminateThread

在CWinThread對象中有線程的句柄,可以使用該句柄強行殺死線程。但是不推薦使用這種方式,當可以正常結束的時候,選擇前兩種方法較好。

檢索線程的退出代碼

若要獲取輔助線程或用戶界面線程的退出代碼,請調用 函數。有關此函數的信息,請參見 Platform SDK。此函數獲取線程(存儲在 CWinThread 對象的 m_hThread 數據成員中)的句柄和 DWORD 的地址。

如果線程仍然是活動的,GetExitCodeThread 將 STILL_ACTIVE 放置在提供的 DWORD 地址中;否則將退出代碼放置在該地址中。

檢索 對象的退出代碼還需要一步。默認情況下,當 CWinThread 線程終止時,刪除該線程對象。這意味着不能訪問 m_hThread 數據成員,因為 CWinThread 對象不再存在。若要避免出現這種情況,請執行以下操作之一:

  • 將 m_bAutoDelete 數據成員設置為 FALSE。這使 CWinThread 對象在線程終止后仍可以繼續存在。然后可以在線程終止后,訪問 m_hThread 數據成員。但是,如果使用此方法,就得銷毀 CWinThread 對象,因為框架不會自動刪除該對象。這是首選方法。

  • 單獨存儲線程的句柄。創建線程后,(使用 ::DuplicateHandle)將其 m_hThread 數據成員復制到其他變量,並通過該變量訪問該成員。這樣,終止后即會自動刪除對象,並且仍然可以找到線程終止的原因。請注意:在可以復制句柄之前,線程不終止。執行此操作的最安全的方式是將 CREATE_SUSPENDED 傳遞到 ,存儲句柄,然后通過調用 繼續執行線程。

任一方法都可以使您確定 CWinThread 對象終止的原因。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM