0

同時に実行するプログラムのインスタンスを 1 つだけにする必要がある状況があります。

これは次のように簡単です。

class OneAtATimePlease
{
  static void Main()
  {
    // Naming a Mutex makes it available computer-wide. Use a name that's
    // unique to your company and application (e.g., include your URL).

    using (var mutex = new Mutex (false, "oreilly.com OneAtATimeDemo"))
    {
      // Wait a few seconds if contended, in case another instance
      // of the program is still in the process of shutting down.

      if (!mutex.WaitOne (TimeSpan.FromSeconds (3), false))
      {
        Console.WriteLine ("Another app instance is running. Bye!");
        return;
      }
      RunProgram();
    }
  }

  static void RunProgram()
  {
    Console.WriteLine ("Running. Press Enter to exit");
    Console.ReadLine();
  }
}

小さな詳細を除いて、新しいプロセスではなく、既存のプロセスを終了する必要があります。

上記のミューテックスを取得した後、既存のプロセスがリッスンできるセマフォを作成しようとしましたが、セマフォを待機させたいため、セマフォが常に通知され、機能しない状況に陥る可能性があります。

誰でもこの問題を解決する方法について良い考えを持っていますか?

4

3 に答える 3

1

シネ、2 つの Mutex ロジックを作成しました。1 つ目は「実行ロック」で、2 つ目は「監視ロック」です。最初のプロセスが監視ロックを取得できない場合、プロセスは終了し、新しいプロセスの実行ロックを解放します。

これが最善の解決策であるかどうかはわかりませんが、フィードバックを歓迎します。

C#:

class Program
    {
        private static string processName = DateTime.Now.ToString("hh.mm.ss");
        private static bool exitProcess;

        private static Mutex firstLock
        {
            get
            {
                return new Mutex(false, "stackoverflow.com/questions/11304052/");
            }
        }

        private static Mutex secondLock
        {
            get
            {
                return new Mutex(false, "stackoverflow.com/questions/11304052/ #2");
            }
        }


        static void Main(string[] args)
        {
            Console.WriteLine(string.Format("Process {0} starting", processName));

            exitProcess = false;

            while (true)
            {
                using (firstLock)
                {
                    Console.WriteLine(string.Format("Process {0} trying to get #1 mutex", processName));

                    if (!firstLock.WaitOne(TimeSpan.FromSeconds(1), false))
                    {
                        Console.WriteLine(string.Format("Process {0} #1 mutex in use, waiting for release", processName));

                        bool killFirstApp = false;

                        while (!killFirstApp)
                        {
                            killFirstApp = LockSecondMutex();
                        }

                        continue;
                    }

                    new Thread(MonitorSecondMutex).Start();

                    RunProgram();

                    firstLock.ReleaseMutex();

                    break;
                }
            }
        }

        static void RunProgram()
        {
            while (!exitProcess)
            {
                Console.WriteLine(string.Format("Process {0} running", processName));

                Thread.Sleep(1000);
            }
        }

        static void MonitorSecondMutex()
        {
            while (true)
            {
                using (secondLock)
                {
                    if (!secondLock.WaitOne(TimeSpan.FromSeconds(2), false))
                    {
                        Console.WriteLine(string.Format("Process {0} lost second mutex. Will now exit.", processName));

                        exitProcess = true;

                        break;
                    }

                    secondLock.ReleaseMutex();
                }

                Thread.Sleep(500);
            }
        }

        static bool LockSecondMutex()
        {
            while (true)
            {
                using (secondLock)
                {
                    if (!secondLock.WaitOne(TimeSpan.FromSeconds(2), false))
                    {
                        continue;
                    }

                    Thread.Sleep(5000);

                    secondLock.ReleaseMutex();
                }

                return true;
            }
        }
    }
于 2012-07-03T05:46:29.457 に答える
1

既存のアプリケーションにシグナルを送信するには、プロセス間通信が必要です。C# については、C# の IPC メカニズム - 使用法とベスト プラクティス を参照してください。

于 2012-07-03T03:29:04.307 に答える
0

ポートなどの限られたシステム全体のリソースへのアクセスを要求することで、これを実行できる可能性があります。アプリケーションは、起動時にソケットを特定のポートにバインドできます。バインドに失敗した場合は、実行中のインスタンスに終了シグナルを送信して、再試行してください。

于 2012-07-03T03:29:47.740 に答える