问题描述:
在监控程序中,设计一监控循环。
头文件.h
HANDLE                                          m_hEventExit;
CWinThread*                                 m_pThread;
构造函数中,创建该事件
m_hEventExit=CreateEvent(NULL,   // 安全
TRUE,  // 手动
FALSE, // 初始化为非信号模式 
_T("Exit Event")  // 线程名称
);
 
在OnButtonThreadStart()中
{
if(!m_pThread)
              {
                            ResetEvent(m_hEventExit);
                            m_ pThread = AfxBeginThread(MonitorThreadFunc, this);
              }
}
MonitorThreadFunc 中需要修改主界面中的控件。
 
这时候如果在OnButtonThreadStop()中
{
              SetEvent(m_hEventExit);
 
              if(m_ pThread!= NULL)
              {
                            TRACE0("The thread is still running.\n");
                            WaitForSingleObject(m_ pThread ->m_hThread, -1);
 
                            delete m_ pThread;
                            m_ pThread = NULL;
              }
}
其中Wait行使主界面进入等待状态,如果这时候工作线程执行完了,可以顺利退出,如果线程此时正在更新界面控件,就会陷入死锁。
 
解决方法:
使用WaitThreadWithHandleMsg函数,可以在等待线程结束的同时响应消息。
为了使用方便,将该函数封装了一下,使用的时候只需要调用一下。
 
 
 int WINAPI WaitThreadWithHandleMsg(HANDLE hEventThread)
int WINAPI WaitThreadWithHandleMsg(HANDLE hEventThread)


 {
{
 HRESULT        hResult = S_FALSE;
    HRESULT        hResult = S_FALSE;
 BOOL        bWait = TRUE;
    BOOL        bWait = TRUE;

 while (bWait)
    while (bWait) 

 
     {
{
 DWORD dwEvt = MsgWaitForMultipleObjects(1,&hEventThread,FALSE,INFINITE,QS_ALLINPUT);
        DWORD dwEvt = MsgWaitForMultipleObjects(1,&hEventThread,FALSE,INFINITE,QS_ALLINPUT);

 switch(dwEvt)
        switch(dwEvt) 

 
         {
{
 case WAIT_OBJECT_0:
        case WAIT_OBJECT_0:        
 bWait = false;
            bWait = false;
 hResult = TRUE;
            hResult = TRUE;
 break;
            break;
 case WAIT_OBJECT_0 + 1:
        case WAIT_OBJECT_0 + 1:        

 
         {
{
 MSG msg;
            MSG msg;
 while(::PeekMessage(&msg, NULL, 0, 0, PM_NOREMOVE))
            while(::PeekMessage(&msg, NULL, 0, 0, PM_NOREMOVE)) 

 
             {
{
 if (WM_CLOSE == msg.message || WM_QUIT == msg.message)
                if (WM_CLOSE == msg.message || WM_QUIT == msg.message) 

 
                 {
{
 bWait = false;
                     bWait = false;
 break;
                     break;
 }
                }
 else
                else 

 
                 {
{
 PeekMessage(&msg, NULL, 0, 0, PM_REMOVE);
                    PeekMessage(&msg, NULL, 0, 0, PM_REMOVE);
 TranslateMessage(&msg);
                    TranslateMessage(&msg);
 DispatchMessage(&msg);
                    DispatchMessage(&msg);
 }
                }
 }
            }
 break;
            break;
 }
        }
 default:  // WAIT_TIMEOUT  WAIT_FAILED
        default:  // WAIT_TIMEOUT  WAIT_FAILED
 bWait = false;
            bWait = false;
 hResult = FALSE;
            hResult = FALSE;
 break;
            break;
 }
        }
 }  // end while
    }  // end while

 return hResult;
    return hResult;
 }
}