我在工程里使用以下函數給線程設置一個方便調試的名稱
typedef struct tagTHREADNAME_INFO
{
DWORD dwType; // must be 0x1000
LPCSTR szName; // pointer to name (in user addr space)
DWORD dwThreadID; // thread ID (-1=caller thread)
DWORD dwFlags; // reserved for future use, must be zero
} THREADNAME_INFO;
void SetThreadName( DWORD dwThreadID, LPCSTR szThreadName )
{
THREADNAME_INFO info;
info.dwType = 0x1000;
info.szName = szThreadName;
info.dwThreadID = dwThreadID;
info.dwFlags = 0;
// Visit http://www.shnenglu.com/sunicdavy for original article
__try
{
RaiseException( 0x406D1388, 0, sizeof(info)/sizeof(DWORD), (DWORD*)&info );
}
__except(EXCEPTION_CONTINUE_EXECUTION)
{
int a =1;
}
}
使用期間一直正常. 隨著工程需求變化,我需要頻繁的將一個被命名的線程創建工作一段時間后馬上結束釋放資源并重新創建,出現了死鎖問題
死鎖總是發生在主線程的Sleep或者是SetEvent兩個WindowsAPI函數. 而且僅僅在Visual Studio 2008里調試時會發生死鎖,脫離調試環境一切正常.
我開始打日志調試, 結果發現死鎖前能夠正常創建線程,但是線程函數根本沒有調用進入.之后就在主線程Sleep處死鎖了.
由于SetThreadName沒有使用正規的API接口,而是使用拋異常方式, 而且線程名稱應該只在Visual Studio IDE中才能顯示, MSDN對RaiseException的描述里也有說明可以使用WaitForDebugEvent接收應用程序拋得異常.因此可以推斷是SetThreadName造成的問題. 屏蔽SetThreadName,測試通過