MFC【17-3】线程和线程同步化

时间:2023-03-09 17:12:17
MFC【17-3】线程和线程同步化

17.3小知识点

17.3.1消息泵

编写一个应用程序,让它响应某菜单命令,画几千个椭圆。

 void CMFC线程View::OnStartDrawing(void)
{
m_bQuit=FALSE;
for(int i=;i<NUMELLIPSES&&!m_bQuit;i++)
{
DrawRandomEllipse();
if(!PeekAndPump())
break;
}
} void CMFC线程View::OnStopDrawing(void)
{
m_bQuit=TRUE;
} bool CMFC线程View::PeekAndPump(void)
{
MSG msg;
while(::PeekMessage(&msg,NULL,,,PM_NOREMOVE)){
if(!AfxGetApp()->PumpMessage()){
::PostQuitMessage();
return FALSE;
}
}
LONG 1Idle=;
while(AfxGetApp()->OnIdle(1Idle++));
return TRUE
}

PeekAndPump在一个消息循环中定制另一个消息循环。它在OnStartDrawing中for语句的循环终点处被调用。如果::PeekMessage指示队列中有消息等待,则PeekAndPump首先调用CWinThread::PumpMessage提取消息和分派消息。如果PumpMessage返回0,则表示提取和分派的最后一个消息是WM_QUIT消息。而因为只有用“主”消息循环提取WM_QUIT消息,应用程序才能结束,所以该消息要求特殊处理。因此如果PumpMessage返回0,PeekAndPump就会把另一个WM_QUIT消息发往队列;如果PeekAndPump返回0,就会把另一个WM_QUIT消息发往队列;如果PeekAndPump返回0,OnStartDrawing中的for语句循环就会失败。如果WM_QUIT消息不提示提前退出,那么通过在返回之前调用应用程序对象的OnIdle函数,PeekAndPump就可以模仿主程序的闲置机制。

17.3.2执行其他进程

Win32执行进程。下列语句执行c:\\WINDOWS\Notepad.exe.

     STARTUPINFO si;
::ZeroMemory(&si,sizeof(STARTUPINFO));
si.cb=sizeof(STARTUPINFO);
PROCESS_INFORMATION pi; if(::CreateProcess(NULL,_T("C:\\Windows\\Notepard"),NULL,NULL,FALSE,NORMAL_PRIORITY_CLASS,
NULL,NULL,&si,&pi)){
::CloseHandle(pi.hThread);
::CloseHandle(pi.hProcess);
}

::CreateProcess是一个通用函数,它获取可执行文件的名字(和路径),然后加载并执行他,如果可执行文件名中的驱动器和目录名被省略,则系统自动在Windows目录,Windows系统目录,当前路径下的所有目录和选中的其他位置中搜索该文件。文件名也可以包含命令行参数,如:

“C:\\Windows\\Notepad C: \\ Windows\\ Desktop\\Ideas.txt"

::CreatProcess将进程的关键信息填充在PROCESS_INFORMATION结构中。相关信息包括:进程句柄(hProcess)和进程中主线程的句柄(hThread)。在进程启动后,要用::CloseHandle关闭这些句柄。如果CreateProcess返回非零值,则意味着进程启动成功。因为Win32是一部启动、一部执行的,所以CreateProcess不必等到进程结束后再返回。如果您希望启动另一个进程,并暂停当前进程知道该进程启动的进程结束,则您可以对该进程句柄调用::WaitForSingleObject

    STARTUPINFO si;
::ZeroMemory(&si,sizeof(STARTUPINFO));
si.cb=sizeof(STARTUPINFO);
PROCESS_INFORMATION pi; if(::CreateProcess(NULL,_T("C:\\Windows\\Notepard"),NULL,NULL,FALSE,NORMAL_PRIORITY_CLASS,
NULL,NULL,&si,&pi)){
::CloseHandle(pi.hThread);
::WaitForSingleObject(pi.hProcess,INFINITE);
::CloseHandle(pi.hProcess);
}

进程和线程一样都有退出代码。如果::WaitForSingleObject没有返回WAIT_FAILED,则可以调用::GetExitCodeProcess获取进程的退出代码。

  有时需要启动进程并等待足够长的时间后,才能确保进程已开始并相应用户输入。例如:如果进程A启动进程B,而进程B又创建了一个窗口,这是,如果进程A要给窗口发送消息,他就不得不等到CreateProcess返回,留给进程B足够的时间创建窗口并开始处理消息。通过Win32::WaitForInputIdle函数可以解决这个问题。

    STARTUPINFO si;
::ZeroMemory(&si,sizeof(STARTUPINFO));
si.cb=sizeof(STARTUPINFO);
PROCESS_INFORMATION pi; if(::CreateProcess(NULL,_T("C:\\Windows\\Notepard"),NULL,NULL,FALSE,NORMAL_PRIORITY_CLASS,
NULL,NULL,&si,&pi)){
::CloseHandle(pi.hThread);
::WaitForInputIdle(pi.hProcess,INFINITE);
::CloseHandle(pi.hProcess);
}

17.3.3文件改变通知

::WaitForSingleObject的HANDLE参数可以是“文件改变通知句柄”。Win32 API包含一个函数::FindFirstChangeNotification,只要给定目录或他的子目录发生了变化,例如文件被重命名或删除,或创建了一个新目录,该函数都能返回一个句柄,通过该句柄您可以启动一个被阻塞的线程。

如果想改善11章中的Wanderer应用程序,使文件系统的而变化难呢过立刻反应在左面或右面的窗格中。为此,最有效的方法是启动一个后台线程,并使它在一个或多个文件改变通知句柄上处于阻塞状态。下面是用来监视驱动器C:的线程的线程函数:

UINT ThreadFunc(LPVOID pParam)
{
HWND hwnd=(HWND)pParam;//Window to notify
HANDLE hChange = ::FindFirstChangeNotification(_T("C:\\"),
TRUE,FILE_NOTIFY_CHANGE_FILE_NAME);// FILE_NOTIFY_CHANGE_DIR_NAME);
if(hChange==INVALID_HANDLE_VALUE){
TRACE(_T("Error:FindFirstChangeNotification failed\n"));
return(UINT) -;
}
while(){
::WaitForSingleObject(hChange,INFINITE);
::PostMessage(hwnd,WM_USER_CHANGE_NOTIFY,,);
::FindNextChangeNotification(hChange);//Reset
}
::FindCloseChangeNotification(hChange);
return ;
}