gpt4 book ai didi

c - 使用 C 的 Windows 进程间通信 (IPC)

转载 作者:可可西里 更新时间:2023-11-01 09:56:08 27 4
gpt4 key购买 nike

我有一个用 C 语言和 Microsoft Visual C++ 编写的旧程序,我需要实现某种“保持事件”,因此我能够将进程间通信接收到一个新程序中,该程序将终止并重新启动如果在过去 5 秒内没有收到消息,则第一个。

问题是我一直在寻找任何 C 语言的 Windows IPC 教程或示例,但我找到的几乎所有内容都是针对 C++ 的。

任何帮助或资源?

编辑:正如@Adriano 在回答中所建议的那样,我正在尝试使用共享内存。但是由于我无法捕获的某种异常,启动程序正在被 Windows 终止。调用 CopyMemory 时发生。

代码如下:

#include "stdafx.h"
#include "windows.h"
#include "iostream"
using namespace std;

int launchMyProcess();
void killMyProcess();
bool checkIfMyProcessIsAlive();

STARTUPINFO sInfo;
PROCESS_INFORMATION pInfo;
HANDLE mappedFile;
LPVOID pSharedMemory;
long lastReceivedBeatTimeStamp;
const int MSECONDS_WITHOUT_BEAT = 500;
const LPTSTR lpCommandLine = "MyProcess.exe configuration.txt";


int main(int argc, char* argv[])
{
mappedFile = CreateFileMapping(INVALID_HANDLE_VALUE, NULL, PAGE_READWRITE, 0, sizeof(int), "Global\\ActivityMonitor");
LPVOID pSharedMemory = MapViewOfFile(mappedFile, FILE_MAP_READ, 0, 0, sizeof(int));
if(!launchMyProcess()){
cout<<"Error creating MyProcess.exe"<<endl;
UnmapViewOfFile(pSharedMemory);
CloseHandle(mappedFile);
return -1;
}
while(true){
Sleep(100);

if(!checkIfMyProcessIsAlive()){
cout<<"Relaunching MyProcess...";
killMyProcess();
if(!launchMyProcess()){
cout<<"Error relaunching MyProcess.exe"<<endl;
UnmapViewOfFile(pSharedMemory);
CloseHandle(mappedFile);
return -1;
}
}
}

UnmapViewOfFile(pSharedMemory);
CloseHandle(mappedFile);
return 0;
}


bool checkIfMyProcessIsAlive()
{
static int volatile latestMagicNumber = 0;
int currentMagicNumber = 0;

CopyMemory(&currentMagicNumber, pSharedMemory, sizeof(int));

if(currentMagicNumber != latestMagicNumber){
latestMagicNumber = currentMagicNumber;
return true;
}
return false;
}

int launchMyProcess()
{
ZeroMemory(&sInfo, sizeof(sInfo));
sInfo.cb = sizeof(sInfo);
ZeroMemory(&pInfo, sizeof(pInfo));

return CreateProcess(NULL, lpCommandLine, NULL, NULL, FALSE, 0, NULL, NULL, &sInfo, &pInfo);
}

void killMyProcess()
{
TerminateProcess(pInfo.hProcess, 0);
CloseHandle(pInfo.hProcess);
CloseHandle(pInfo.hThread);
Sleep(3000);
}

最佳答案

如果您的旧 C 应用程序有一个消息泵(因为它有一个 UI),那么检查它是否存在的最简单方法可能是 IsHungAppWindow()功能,Windows 会为您完成这些工作。

如果这不是您的情况并且您需要 IPC,则有很多选择,这取决于您要使用哪种 IPC 机制。这里我只列出一些资源。

IPC 技术概述:http://msdn.microsoft.com/en-us/library/windows/desktop/aa365574(v=vs.85).aspx

一些例子:

编辑
我认为一个小例子比大量的文字更能说明问题。在此示例中,我将使用共享内存,但您可以使用您喜欢的任何东西(而且您会觉得更舒服)。未经测试,请仅作引用。

应首先启动 MONITOR 进程。

VOID CALLBACK CheckItIsAlive(PVOID lpParam, BOOLEAN TimerOrWaitFired)
{
static int volatile latestMagicNumber = 0;

int currentMagicNumber = 0;
CopyMemory(&currentMagicNumber, lpParam, sizeof(int));

if (currentMagicNumber != latestMagicNumber)
latestMagicNumber = currentMagicNumber;
else
{
// Do something, it's hanged
}
}

void main()
{
// Shared memory used to communicate with the other process
HANDLE mappedFile = CreateFileMapping(INVALID_HANDLE_VALUE,
NULL, PAGE_READWRITE, 0, sizeof(int), "Global\\MyActivityMonitor");

LPVOID pSharedMemory = MapViewOfFile(mappedFile,
FILE_MAP_READ, 0, 0, sizeof(int));

// Thread used to check activity
HANDLE queue = CreateTimerQueue();
HANDLE hTimer = NULL;
CreateTimerQueueTimer(&hTimer, queue,
(WAITORTIMERCALLBACK)CheckItIsAlive, pSharedMemory,
0, 5000, WT_EXECUTEDEFAULT);

// Do your work here...

// Clean up
DeleteTimerQueue(queue);

UnmapViewOfFile(pSharedMemory);
CloseHandle(mappedFile);
}

MONITORED 进程,它将向 Monitor 进程发送其事件信号。

VOID CALLBACK NotifyImAlive(PVOID lpParam, BOOLEAN TimerOrWaitFired)
{
static int volatile counter = 1;

int tick = counter++;
CopyMemory(lpParam, &tick, sizeof(int));
}

void main()
{
// Shared memory used to communicate with the other process
HANDLE mappedFile = OpenFileMapping(FILE_MAP_ALL_ACCESS, FALSE,
"Global\\MyActivityMonitor");

LPVOID pSharedMemory = MapViewOfFile(mappedFile,
FILE_MAP_WRITE, 0, 0, sizeof(int));

// Thread used to signal activity
HANDLE queue = CreateTimerQueue();
HANDLE hTimer = NULL;
CreateTimerQueueTimer(&hTimer, queue,
(WAITORTIMERCALLBACK)NotifyImAlive, pSharedMemory,
0, 5000, WT_EXECUTEINTIMERTHREAD);

// Do your work here...

// Clean up
DeleteTimerQueue(queue);

UnmapViewOfFile(pSharedMemory);
CloseHandle(mappedFile);
}

共享内存是一种非常轻量级的资源,您可以使用任何您喜欢的计时器(如果计时不是严格要求,您可以进行某种空闲处理。我个人喜欢这个,因为您不需要锁定任何线程,并且您可能有一个空闲时间处理线程)。

从 Windows 2000 开始支持定时器功能,确保 _WIN32_WINNT 宏定义为 0x0500(或更多)。

附录
我没有在列表中提到,因为它们只存在于较新版本的操作系统中,但您甚至可以使用 condition variables . Windows 8 将支持一个非常有用的 WaitOnAddress功能,但它仍然是 future ,所以我认为你不能使用它。

关于c - 使用 C 的 Windows 进程间通信 (IPC),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10535313/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com