Вопрос по console-application, complexity-theory, .net – Событие выхода из консольного приложения .NET

53

Есть ли в .NET метод, например событие, для определения момента выхода из консольного приложения? Мне нужно очистить некоторые потоки и COM-объекты.

Я запускаю цикл сообщений без формы из консольного приложения. Компонент DCOM, который я использую, кажется, требует, чтобы приложение качало сообщения.

Я попытался добавить обработчик для Process.GetCurrentProcess.Exited и Process.GetCurrentProcess.Disposed.

Я также попытался добавить обработчик к событиям Application.ApplicationExit и Application.ThreadExit, но они не запускаются. Возможно, это потому, что я не использую форму.

Событие Process.GetCurrentProcess (). Exited должно возникать, если вы сначала установили Process.EnableRaisingEvents = true; в противном случае он не сработает. Triynko
@Triynko, как это? Process.Exited запускается после завершения процесса. И если собственное приложение закрывается, то код вообще не выполняется. Довольно бесполезный имо .. nawfal
Полный рабочий пример в конце этого поста в C # JJ_Coder4Hire
возможный дубликатCapture console exit C# nawfal

Ваш Ответ

6   ответов
23

Вот полное, очень простое решение .Net, которое работает во всех версиях Windows. Просто вставьте его в новый проект, запустите его и попробуйте CTRL-C, чтобы посмотреть, как он это обрабатывает:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.InteropServices;
using System.Text;
using System.Threading;

namespace TestTrapCtrlC{
    public class Program{
        static bool exitSystem = false;

        #region Trap application termination
        [DllImport("Kernel32")]
        private static extern bool SetConsoleCtrlHandler(EventHandler handler, bool add);

        private delegate bool EventHandler(CtrlType sig);
        static EventHandler _handler;

        enum CtrlType {
         CTRL_C_EVENT = 0,
         CTRL_BREAK_EVENT = 1,
         CTRL_CLOSE_EVENT = 2,
         CTRL_LOGOFF_EVENT = 5,
         CTRL_SHUTDOWN_EVENT = 6
         }

        private static bool Handler(CtrlType sig) {
            Console.WriteLine("Exiting system due to external CTRL-C, or process kill, or shutdown");

            //do your cleanup here
            Thread.Sleep(5000); //simulate some cleanup delay

            Console.WriteLine("Cleanup complete");

            //allow main to run off
             exitSystem = true;

            //shutdown right away so there are no lingering threads
            Environment.Exit(-1);

            return true;
        }
        #endregion

        static void Main(string[] args) {
            // Some biolerplate to react to close window event, CTRL-C, kill, etc
            _handler += new EventHandler(Handler);
            SetConsoleCtrlHandler(_handler, true);

            //start your multi threaded program here
            Program p = new Program();
            p.Start();

            //hold the console so it doesn’t run off the end
            while(!exitSystem) {
                Thread.Sleep(500);
            }
        }

        public void Start() {
            // start a thread and start doing some processing
            Console.WriteLine("Thread started, processing..");
        }
    }
 }
код не работает в windows xp sp3. если мы закроем приложение из диспетчера задач.
Кажется, это лучшее решение C #, которое работает для всех сценариев выхода! Работает отлично! Спасибо :)
Будет ли этот код работать в Linux в консольном приложении .NET Core? Я не пробовал, но "Kernel32" кажется не переносимым для меня ...
7

Для случая CTRL + C вы можете использовать это:

// Tell the system console to handle CTRL+C by calling our method that
// gracefully shuts down.
Console.CancelKeyPress += new ConsoleCancelEventHandler(Console_CancelKeyPress);


static void Console_CancelKeyPress(object sender, ConsoleCancelEventArgs e)
{
            Console.WriteLine("Shutting down...");
            // Cleanup here
            System.Threading.Thread.Sleep(750);
}
Это обрабатывает завершение работы выхода из системы Ctrl + C и т. Д ...? user79755
Это работало нормально для меня при запуске консольного приложения.
Это не работает. Я попробовал это.
Пробовал, событие никогда не срабатывает. То же, что и другие события. user79755
16

Приложение представляет собой сервер, который просто запускается до тех пор, пока система не выключится или не получит Ctrl + C или окно консоли не будет закрыто.

Из-за необычного характера заявки «изящно» не представляется возможным выход. (Возможно, я мог бы написать другое приложение, которое отправило бы сообщение «завершение работы сервера», но это было бы излишним для одного приложения и все же недостаточным для определенных обстоятельств, например, когда сервер (фактическая ОС) фактически выключается.)

Из-за этих обстоятельств я добавил & quot;ConsoleCtrlHandler& Quot; где я останавливаю свои потоки и очищаю мои COM-объекты и т.д ...


Public Declare Auto Function SetConsoleCtrlHandler Lib "kernel32.dll" (ByVal Handler As HandlerRoutine, ByVal Add As Boolean) As Boolean

Public Delegate Function HandlerRoutine(ByVal CtrlType As CtrlTypes) As Boolean

Public Enum CtrlTypes
  CTRL_C_EVENT = 0
  CTRL_BREAK_EVENT
  CTRL_CLOSE_EVENT
  CTRL_LOGOFF_EVENT = 5
  CTRL_SHUTDOWN_EVENT
End Enum

Public Function ControlHandler(ByVal ctrlType As CtrlTypes) As Boolean
.
.clean up code here
.
End Function

Public Sub Main()
.
.
.
SetConsoleCtrlHandler(New HandlerRoutine(AddressOf ControlHandler), True)
.
.
End Sub

Эта настройка, кажется, работает отлично. Вотссылка на сайт к некоторому коду C # для того же самого.

@ dan-gph Программа завершается, когда заканчивается основной поток. Добавьте что-то вроде // удержания консоли, чтобы она не запускалась до конца, пока (! ExitSystem) {Thread.Sleep (500); }
Похоже, что для обработчика есть ограничение в 3 секунды. Если вы не успеете вовремя, вас уволят. Это включает в себя, если вы сидите на точке останова в отладчике!
Это работает, но полный рабочий образец находится ниже в C #
Спасибо, это помогло мне :) +1 Вы должны пометить свой ответ как ответ.
Этот код не запускается, если вы используете End Process в диспетчере задач.
1

Если вы используете консольное приложение и отправляете сообщения, можете ли вы использовать сообщение WM_QUIT?

Мне тоже интересно об этом. Я полагаюcould может случиться так, что Windows не отправляет сообщение в консольное приложение, но мне это кажется странным. Следует подумать, что требовалось бы иметь насос сообщений, а не то, имеет ли он графический интерфейс ...
-1

Как хороший пример может стоить тогоперейти к этому проекту и посмотреть, как обрабатывать выходящие процессы грамматически или в этом фрагменте изВМ нашел здесь

                ConsoleOutputStream = new ObservableCollection<string>();

                var startInfo = new ProcessStartInfo(FilePath)
                {
                    WorkingDirectory = RootFolderPath,
                    Arguments = StartingArguments,
                    RedirectStandardOutput = true,
                    UseShellExecute = false,
                    CreateNoWindow = true
                };
                ConsoleProcess = new Process {StartInfo = startInfo};

                ConsoleProcess.EnableRaisingEvents = true;

                ConsoleProcess.OutputDataReceived += (sender, args) =>
                {
                    App.Current.Dispatcher.Invoke((System.Action) delegate
                    {
                        ConsoleOutputStream.Insert(0, args.Data);
                        //ConsoleOutputStream.Add(args.Data);
                    });
                };
                ConsoleProcess.Exited += (sender, args) =>
                {
                    InProgress = false;
                };

                ConsoleProcess.Start();
                ConsoleProcess.BeginOutputReadLine();
        }
    }

    private void RegisterProcessWatcher()
    {
        startWatch = new ManagementEventWatcher(
            new WqlEventQuery($"SELECT * FROM Win32_ProcessStartTrace where ProcessName = '{FileName}'"));
        startWatch.EventArrived += new EventArrivedEventHandler(startProcessWatch_EventArrived);

        stopWatch = new ManagementEventWatcher(
            new WqlEventQuery($"SELECT * FROM Win32_ProcessStopTrace where ProcessName = '{FileName}'"));
        stopWatch.EventArrived += new EventArrivedEventHandler(stopProcessWatch_EventArrived);
    }

    private void stopProcessWatch_EventArrived(object sender, EventArrivedEventArgs e)
    {
        InProgress = false;
    }

    private void startProcessWatch_EventArrived(object sender, EventArrivedEventArgs e)
    {
        InProgress = true;
    }
55

Вы можете использоватьProcessExit событиеAppDomain:

class Program
{
    static void Main(string[] args)
    {
        AppDomain.CurrentDomain.ProcessExit += new EventHandler(CurrentDomain_ProcessExit);           
        // do some work

    }

    static void CurrentDomain_ProcessExit(object sender, EventArgs e)
    {
        Console.WriteLine("exit");
    }
}

Update

Вот полный пример программы с пустым «насосом сообщений» работает в отдельном потоке, что позволяет пользователю вводить команду выхода в консоли, чтобы корректно закрыть приложение. После цикла в MessagePump вы, вероятно, захотите очистить ресурсы, используемые потоком. Там лучше сделать это, чем в ProcessExit по нескольким причинам:

Avoid cross-threading problems; if external COM objects were created on the MessagePump thread, it's easier to deal with them there. There is a time limit on ProcessExit (3 seconds by default), so if cleaning up is time consuming, it may fail if pefromed within that event handler.

Вот код:

class Program
{
    private static bool _quitRequested = false;
    private static object _syncLock = new object();
    private static AutoResetEvent _waitHandle = new AutoResetEvent(false);

    static void Main(string[] args)
    {
        AppDomain.CurrentDomain.ProcessExit += new EventHandler(CurrentDomain_ProcessExit);
        // start the message pumping thread
        Thread msgThread = new Thread(MessagePump);
        msgThread.Start();
        // read input to detect "quit" command
        string command = string.Empty;
        do
        {
            command = Console.ReadLine();
        } while (!command.Equals("quit", StringComparison.InvariantCultureIgnoreCase));
        // signal that we want to quit
        SetQuitRequested();
        // wait until the message pump says it's done
        _waitHandle.WaitOne();
        // perform any additional cleanup, logging or whatever
    }

    private static void SetQuitRequested()
    {
        lock (_syncLock)
        {
            _quitRequested = true;
        }
    }

    private static void MessagePump()
    {
        do
        {
            // act on messages
        } while (!_quitRequested);
        _waitHandle.Set();
    }

    static void CurrentDomain_ProcessExit(object sender, EventArgs e)
    {
        Console.WriteLine("exit");
    }
}
Это не работает. вставил его в новый консольный проект, разрешил зависимости, установил точку останова для CurrentDomain_ProcessExit, и он не срабатывает при нажатии CTRL-C или при нажатии X на окне. Смотрите мой полный рабочий ответ ниже
Ты серьезно? Конечно, когда любое приложение Windows.Forms закрывается пользователем, закрывающим основную форму, у вас есть шанс выполнить некоторую очистку! Я реализую пакет, и совершенно недопустимо просто умирать от этого процесса, не имея возможности даже зарегистрировать тот факт, что задание было прервано ...
Как завершается ваш процесс? Вы называете какой-либо код, который делает его выход?
Спасибо за предложение, Фредрик. Это событие, похоже, не сработало. user79755
Ах .. это имеет смысл; вы не выходите из приложения, вы убиваете его грубой силой. Там нет события, которое поднимет это. Вам нужно реализовать способ изящного выхода из приложения.

Похожие вопросы