上家面试公司项目经理问我一个问题:“重复运行同一个程序时再次弹出显示怎么实现?”
当时我的回答是CreateMutex,然后RegisterWindowMessage注册一个专用消息,下次检测到该消息就直接PostMessage让它弹出就行了。
那个经理很轻蔑地说,为什么这么复杂?直接CreateMutex,然后FindWindow不就行了吗?
当晚把我问晕了,但我知道他一定想得太简单了,但我想不起来原因了。
今天再看了一下当时代码,那二逼经理说得是没错,但仅于QT环境下,因为在MFC下默认是不能修改窗口类名的(一个对话框一定是32770),那么在FindWindow的时候就只能通过窗口名来查,而窗口名这个东西是最容易【重复】的,所以这里FindWindow根本就行不通,因此当时才想到了使用RegisterWindowMessage这个办法。
回敬一句那个经理,简单你妈B,好好认真写一个MFC登录程序再来反驳我的观点,别TM以为用了几年QT就觉得自己可以当项目经理了,连FindWindow依赖的参数都没弄清楚还在那儿瞎BB,搞得我很业余一样,真你MA操蛋。
bool ReOpen()
{
CString main_guid(“guidxxx”)
HANDLE m_hMutex = CreateMutex(NULL, TRUE, main_guid); // 重复打开判断
UINT m_uiExist = RegisterWindowMessage(main_guid); // 注册消息,重复打开时响应。
if(GetLastError()==ERROR_ALREADY_EXISTS)
{
PostMessage(HWND_BROADCAST, m_uiExist, 0, 0); // 显示已运行的程序。
return true;
}
return false;
}
然后在需要弹出的窗口中的WindowProc处理m_uiExist消息:
LRESULT CDlg::WindowProc(UINT message, WPARAM wParam, LPARAM lParam)
{
if (message == m_uiExist) // 重复打开时,窗口置顶。
{
// If so, does it have any popups?
CWnd* pWndChild = GetLastActivePopup();
// If iconic, restore the main window
if (IsIconic())
ShowWindow(SW_RESTORE);
// Bring the main window or its popup to the foreground
pWndChild->SetForegroundWindow();
}
return CDialogEx::WindowProc(message, wParam, lParam);
}