检测程序运行实例个数

来源:互联网 发布:mac 个人文件夹改名 编辑:程序博客网 时间:2024/05/22 13:43
共享数据段 方式

在.cpp文件开头加上以下几行:

#pragma data_seg("Shared")

int volatile g_lAppInstance =0;

#pragma data_seg()

#pragma comment(linker,"/section:Shared,RWS")

我们来看一下上面的内容:

第一句 #pragma data_seg("Shared") 创建一个称为Shared 的新节。

第二句 int volatile g_lAppInstance =0 将 g_lAppInstance 放入Shared节中。注意此时只有将g_lAppInstance初始化,编译器才会将其放入Shared节中,否则,将放入Shared以外的节。(实际上Visual C++ 编译器提供了一个allocate 说明符,使我们可以将数据放到任何节中。)

第三句指示编译器Shared 节结束。

#pragma comment(linker,"/section:Shared,RWS")

这一句,我们使编译链接器知道我们的Shared节具有读,写,共享的属性。这是我们实现互斥运行的关键。这样我们就可以在应用程序之间的多个实例之间共享g_lAppInstance 变量。

在InitInstance() 函数中加入:

if(++g_lAppInstance>1)

{

AfxMessageBox("程序已经运行!");

return FALSE;

}

以上代码的作用是在程序是开始对g_lAppInstancd 加1 ,如果发现其值大于1,那么显示Message Box ,并返回FALSE。(注意:在MFC中如果InitInstance返回FALSE,程序将不会被运行!)

#pragma data_seg()一般用于DLL中。也就是说,在DLL中定义一个共享的,有名字的数据段。最关键的是:这个数据段中的全局变量可以被多个进程共享。否则多个进程之间无法共享DLL中的全局变量。

共享数据必须初始化,否则微软编译器会把没有初始化的数据放到.BSS段中,从而导致多个进程之间的共享行为失败。

互斥量方式,只运行一个实例,并激活已运行的程序

BOOL C***App::InitInstance()

{

//创建进程互斥体

CString appName = AfxGetApp()->m_pszAppName;

HANDLE hMutex = CreateMutex(NULL,TRUE, appName);

if (hMutex == NULL)

{

return FALSE;

}

//如果程序已经存在并且正在运行

if (GetLastError() == ERROR_ALREADY_EXISTS)

{

HWND hProgramWnd = ::FindWindow(NULL, appName);

if (hProgramWnd)

{

// 主窗口已最小化,则恢复其大小

if (::IsIconic(hProgramWnd))

{

::ShowWindow(hProgramWnd, SW_RESTORE);

}

// 将主窗激活

::SetForegroundWindow(hProgramWnd);

// 将主窗的对话框激活

::SetForegroundWindow(::GetLastActivePopup(hProgramWnd));

}

//关闭进程互斥体

CloseHandle(hMutex);

hMutex = NULL;

return FALSE;

}

……

}

原创粉丝点击