MFC主线程使用WaitForSingleObject阻塞的问题 http://blog.csdn.net/sysprogram/article/details/17383455

来源:互联网 发布:php get参数加密 编辑:程序博客网 时间:2024/05/23 01:18

MFC主线程使用WaitForSingleObject阻塞的问题

 2339人阅读 评论(1) 收藏 举报
 分类:

在MFC程序的主线程中如果使用WaitForSingleObject等线子线程,而子线程里使用了有关于消息的函数,比如SetWindowText,InsertItem,SetItemText这些函数,就有会导致主主线程阻塞问题,看看这段代码。

[cpp] view plaincopy在CODE上查看代码片派生到我的代码片
  1. DWORD WINAPI ThreadProc(  
  2.                         LPVOID lpParameter   // thread data  
  3.                         )  
  4. {  
  5.     CMFCWaitThreadDlg *pDlg = (CMFCWaitThreadDlg*)lpParameter;  
  6.   
  7.     for (int i=0;i<=10;i++)  
  8.     {  
  9.         OutputDebugStringA("test");  
  10.         Sleep(2000);  
  11.         pDlg->SetWindowText(_T("HEHE"));  
  12.           
  13.     }  
  14.     return true;  
  15. }  
  16. void CMFCWaitThreadDlg::OnBnClickedOk()  
  17. {  
  18.     // TODO: 在此添加控件通知处理程序代码  
  19.     DWORD dwID;  
  20.     HANDLE hThread;  
  21.     hThread = CreateThread(0,0,ThreadProc,this,0,&dwID);  
  22.   
  23.     DWORD dwRet = WaitForSingleObject(hThread,INFINITE);  
  24.     if (dwRet == WAIT_OBJECT_0)  
  25.     {  
  26.         MessageBoxA(0,"Thread exit",0,0);  
  27.     }  
  28.   
  29. }  


主线程出现阻塞问题的原因是,因为SetWindowText调用了SendMessage,借助了消息循环,然后主线程又Waitxxxxx,必然会阻塞,最终导致工作线程Crash掉了。还好微软提供了另一个函数MsgWaitForMultipleObjects,用这个就能搞定了,看看下面这段代码,把WaitForSingleObject那一段给替换了,跑一跑试试,就没问题了。

[cpp] view plaincopy在CODE上查看代码片派生到我的代码片
  1. DWORD WINAPI ThreadProc(  
  2.                         LPVOID lpParameter   // thread data  
  3.                         )  
  4. {  
  5.     CMFCWaitThreadDlg *pDlg = (CMFCWaitThreadDlg*)lpParameter;  
  6.   
  7.     for (int i=0;i<=10;i++)  
  8.     {  
  9.         OutputDebugStringA("test");  
  10.         Sleep(2000);  
  11.         pDlg->SetWindowText(_T("HEHE"));  
  12.           
  13.     }  
  14.     return true;  
  15. }  
  16. void CMFCWaitThreadDlg::OnBnClickedOk()  
  17. {  
  18.     // TODO: 在此添加控件通知处理程序代码  
  19.     DWORD dwID;  
  20.     HANDLE hThread;  
  21.     hThread = CreateThread(0,0,ThreadProc,this,0,&dwID);  
  22.   
  23.     while(TRUE)  
  24.     {  
  25.   
  26.         DWORD result ;   
  27.         MSG msg ;   
  28.   
  29.         result = MsgWaitForMultipleObjects(1, &hThread,   
  30.             FALSE, INFINITE, QS_ALLINPUT);   
  31.   
  32.         if (result == (WAIT_OBJECT_0))  
  33.         {  
  34.             break;  
  35.         }   
  36.         else   
  37.         {   
  38.             PeekMessage(&msg, NULL, 0, 0, PM_REMOVE);  
  39.             DispatchMessage(&msg);   
  40.         }   
  41.     }  
  42.   
  43.     MessageBoxA(0,"Thread exit",0,0);  
  44.   
  45. }  
0 0