[C#] 如何在C#控制台應用程序中捕獲ctrl-c


Answers

Console.CancelKeyPress事件用於此。 這是如何使用它的:

public static void Main(string[] args)
{
    Console.CancelKeyPress += delegate {
        // call methods to clean up
    };

    while (true) {}
}

當用戶按下Ctrl + C時,委託中的代碼將運行並退出程序。 這使您可以通過調用必需的方法來執行清理。 請注意,代理執行後沒有代碼。

還有其他情況下,這不會削減它。 例如,如果程序當前正在執行無法立即停止的重要計算。 在這種情況下,正確的策略可能是告訴程序在計算完成後退出。 下面的代碼給出了一個如何實現它的例子:

class MainClass
{
    private static bool keepRunning = true;

    public static void Main(string[] args)
    {
        Console.CancelKeyPress += delegate(object sender, ConsoleCancelEventArgs e) {
            e.Cancel = true;
            MainClass.keepRunning = false;
        };

        while (MainClass.keepRunning) {
            // Do your work in here, in small chunks.
            // If you literally just want to wait until ctrl-c,
            // not doing anything, see the answer using set-reset events.
        }
        Console.WriteLine("exited gracefully");
    }
}

這段代碼和第一個例子的區別在於e.Cancel被設置為true,這意味著e.Cancel之後的執行會繼續。 如果運行,程序將等待用戶按下Ctrl + C。發生這種情況時, keepRunning變量會更改導致while循環退出的值。 這是使程序正常退出的一種方法。

Question

我希望能夠在C#控制台應用程序中捕獲CTRL + C ,以便在退出之前執行一些清理。 這樣做的最好方法是什麼?




Console.TreatControlCAsInput = true; 已經為我工作。




這是一個完整的工作示例。 粘貼到空的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..");
        }
    }
}