Kill child process when parent process is killed

前端 未结 15 2464
轻奢々
轻奢々 2020-11-22 05:59

I\'m creating new processes using System.Diagnostics.Process class from my application.

I want this processes to be killed when/if my application has cr

15条回答
  •  旧时难觅i
    2020-11-22 06:44

    Here's an alternative that may work for some when you have control of the code the child process runs. The benefit of this approach is it doesn't require any native Windows calls.

    The basic idea is to redirect the child's standard input to a stream whose other end is connected to the parent, and use that stream to detect when the parent has gone away. When you use System.Diagnostics.Process to start the child, it's easy to ensure its standard input is redirected:

    Process childProcess = new Process();
    childProcess.StartInfo = new ProcessStartInfo("pathToConsoleModeApp.exe");
    childProcess.StartInfo.RedirectStandardInput = true;
    
    childProcess.StartInfo.CreateNoWindow = true; // no sense showing an empty black console window which the user can't input into
    

    And then, on the child process, take advantage of the fact that Reads from the standard input stream will always return with at least 1 byte until the stream is closed, when they will start returning 0 bytes. An outline of the way I ended up doing this is below; my way also uses a message pump to keep the main thread available for things other than watching standard in, but this general approach could be used without message pumps too.

    using System;
    using System.IO;
    using System.Threading;
    using System.Windows.Forms;
    
    static int Main()
    {
        Application.Run(new MyApplicationContext());
        return 0;
    }
    
    public class MyApplicationContext : ApplicationContext
    {
        private SynchronizationContext _mainThreadMessageQueue = null;
        private Stream _stdInput;
    
        public MyApplicationContext()
        {
            _stdInput = Console.OpenStandardInput();
    
            // feel free to use a better way to post to the message loop from here if you know one ;)    
            System.Windows.Forms.Timer handoffToMessageLoopTimer = new System.Windows.Forms.Timer();
            handoffToMessageLoopTimer.Interval = 1;
            handoffToMessageLoopTimer.Tick += new EventHandler((obj, eArgs) => { PostMessageLoopInitialization(handoffToMessageLoopTimer); });
            handoffToMessageLoopTimer.Start();
        }
    
        private void PostMessageLoopInitialization(System.Windows.Forms.Timer t)
        {
            if (_mainThreadMessageQueue == null)
            {
                t.Stop();
                _mainThreadMessageQueue = SynchronizationContext.Current;
            }
    
            // constantly monitor standard input on a background thread that will
            // signal the main thread when stuff happens.
            BeginMonitoringStdIn(null);
    
            // start up your application's real work here
        }
    
        private void BeginMonitoringStdIn(object state)
        {
            if (SynchronizationContext.Current == _mainThreadMessageQueue)
            {
                // we're already running on the main thread - proceed.
                var buffer = new byte[128];
    
                _stdInput.BeginRead(buffer, 0, buffer.Length, (asyncResult) =>
                    {
                        int amtRead = _stdInput.EndRead(asyncResult);
    
                        if (amtRead == 0)
                        {
                            _mainThreadMessageQueue.Post(new SendOrPostCallback(ApplicationTeardown), null);
                        }
                        else
                        {
                            BeginMonitoringStdIn(null);
                        }
                    }, null);
            }
            else
            {
                // not invoked from the main thread - dispatch another call to this method on the main thread and return
                _mainThreadMessageQueue.Post(new SendOrPostCallback(BeginMonitoringStdIn), null);
            }
        }
    
        private void ApplicationTeardown(object state)
        {
            // tear down your application gracefully here
            _stdInput.Close();
    
            this.ExitThread();
        }
    }
    

    Caveats to this approach:

    1. the actual child .exe that is launched must be a console application so it remains attached to stdin/out/err. As in the above example, I easily adapted my existing application that used a message pump (but didn't show a GUI) by just creating a tiny console project that referenced the existing project, instantiating my application context and calling Application.Run() inside the Main method of the console .exe.

    2. Technically, this merely signals the child process when the parent exits, so it will work whether the parent process exited normally or crashed, but its still up to the child processes to perform its own shutdown. This may or may not be what you want...

提交回复
热议问题