2

Microsoft Visual C++ を使用して 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);
    }
4

3 に答える 3

2

OP とさまざまなコメントから、アプリケーションがハングしているかどうかを判断することが主な目的であるかのように聞こえます。別のアプリケーションで監視できるある種の「ハートビート」を作成するかなり簡単な方法は、共有メモリまたは名前付きセマフォのいずれかです。

1 つのプロセスでCreateFileMappingMapViewOfFileを使用して共有メモリを作成し、別のプロセスで MapViewOfFile を使用してそのポインタを取得できます。整数のサイズになるように作成した場合、キープアライブの簡単な方法は、プロセスにメモリ内の値を数秒ごとにインクリメントさせることです。他のプロセスは、数秒ごとにそれを読み取って、変更されていることを確認できます。

名前付きセマフォ ( CreateSemaphoreおよびOpenSemaphore ) を使用すると、基本的に同じことができます。監視対象のアプリに定期的にシグナルを送信させ、モニターを待機させて、シグナルが送信されたことを確認します。

于 2012-05-10T15:01:30.830 に答える
2

古い 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 (またはそれ以上) で定義されていることを確認してください。

補遺
は、OSの新しいバージョンにのみ存在するため、リストには記載しませんでしたが、条件変数を使用することもできます。Windows 8 は非常に便利なWaitOnAddress関数をサポートしますが、まだ未来なので、使用できないと思います。

于 2012-05-10T14:01:54.033 に答える
1

これは、あなたが扱っているプラ​​ットフォームに慣れていないために、非常に長い道のりを歩んでいる別のケースのようです.

コメントで言うように、プログラムが生きいるかどうかを知る必要があるだけで、プログラムを強制終了できる場合は、リモートでIPCを必要としません。

監視したいプログラムの最初に:

HANDLE hMutex = CreateMutex(NULL, FALSE, _T("MyMagicKey"));
WaitForSingleObject(hMutex, INFINITE);

「ウォッチドッグ」プログラムでは、次のように他のユーティリティが生きているかどうかを確認します。

HANDLE hMutex = OpenMutex(SYNCHRONIZE, FALSE, _T("MyMagicKey"));
if (hMutex == NULL && GetLastError() == ERROR_FILE_NOT_FOUND)
   //The app being watched is already dead
else
   //whatever you want

同様に適用できる (またはより優れた) 解決策が他にも 6 つほどあります。監視対象のアプリを作成したのがウォッチドッグだけである場合は、送信元(または)HANDLEを待つことができます。CreateProcessShellExecuteEx

于 2012-05-10T14:15:39.590 に答える