c# - 如何取消PC的关机事件?

标签 c# winforms systemevent

我正在制作一个应用程序,我想在系统关闭之前在其中执行一些数据库查询。我正在使用这段代码 -

static void SystemEvents_SessionEnding(object sender, SessionEndingEventArgs e)
{
        e.Cancel = true;
        MessageBox.Show("Shut down canceled");
}

我确实执行了这个应用程序并试图关闭系统并且这个代码也捕获了关闭事件但是问题是在显示消息框之后它也显示了这个屏幕 - [我无法发布图像,因为我没有”没有 10 分。]

它显示了停止系统关闭的我的应用程序的名称,它还提供了“强制关闭按钮”,我不希望显示此屏幕,因为用户可以在执行完成之前强制关闭系统我的查询。

需要专家的建议,在此先感谢。

最佳答案

简短可靠的答案:

在任何最新的 Windows 版本上,您可以尝试取消关机,但 Windows 可能决定忽略您;可悲的是,这是设计使然。如果您绝对必须在进程终止之前完成操作,那么执行此操作的正确位置是在 SessionEnded 处理程序中。如果您有必须在您的进程终止之前完成的任务,则您不得从您的 SessionEnded 处理程序返回,直到您完成所有工作(因此您的查询等已完成)。 )

因此,不是(或者,如果您愿意,也可以)处理 SessionEnding,而是处理 SessionEnded 并且您在那里工作吗:

static void SystemEvents_SessionEnded(object sender, SessionEndedEventArgs e)
{
    WaitForQueriesToFinishOrSaveState(); // new code
}

如何实现等待将取决于您的应用程序;如果您需要重新运行查询,您可以在其中执行它们,或者您可能需要 Thread.Join() 或以其他方式等待后台任务完成;但它必须是阻塞等待(因此您需要在完成之前不要从函数返回)。

因为您不能完全停止关闭,在这种情况下尝试取消可能没有什么意义,所以我建议根本不要在 SessionEnding 中设置 e.Cancel。在较旧的 Windows 版本上,这更有意义,不幸的是,现在意义不大了。

也是the API docs推荐的不是在 SessionEnding(包括消息框)中做任何重要的工作,而是设置您需要立即返回的任何标志,然后在 SessionEnded 中做工作。 (未经证实:我怀疑如果你没有足够快地返回,这可能会加速用户出现“程序行为不端,你想杀死它们”屏幕,因为 Windows 认为你没有玩再好不过了。)

幕后花絮:

设置 e.Cancel 向 Windows 表明您希望 session 不要结束;但用户仍然可以查看; Windows 可能会出于任何它认为相关的原因而决定忽略您的请求。这就是 cookies 崩溃的方式。您可能会发现适用于特定情况的 hack,但没有任何 API 或方法是 Microsoft 批准的,因此现在和将来可能始终如一地起作用。

在幕后,Windows 正在向您的进程的窗口发送 WM_QUERYENDSESSION 消息,.NET 会为您接收该消息并将其转换为 SessionEnding 事件)并将您的取消(或未取消)传递回 Windows;如果您不取消则返回 TRUE,如果您取消则返回 FALSE。

在此之后,Windows 会查看所有进程的请求,并且根据关闭的原因和其他因素,尽管有此类请求,Windows 仍可能决定继续执行。如果进程不合作,它也可能会提醒用户并让他们选择终止进程。

无论您是否处理 WM_QUERYENDSESSION (SessionEnding),您总是有最后一次清理的机会:您收到一条 WM_ENDSESSION 消息(转换为 SessionEnded)。棘手的部分是您必须在所有 SessionEnded 处理程序返回之前完成所有重要任务!

一旦 Windows 从其 WM_ENDSESSION (SessionEnded) 调用中收到反馈,就您的应用程序的生命周期而言,所有赌注都将取消,Windows 可以随时终止您的进程。

Raymond Chen 最近非常专业地对此进行了介绍。

http://blogs.msdn.com/b/oldnewthing/archive/2013/06/27/10429232.aspx

作为脚注,SystemEvents.SessionEnded 是一种便利;如果你有一个顶级应用程序窗口,你可以完全绕过它并通过以下方式实现相同的目的:

protected override void WndProc(ref Message m)
{
    if (m.Msg == 0x16) // WM_ENDSESSION
    {
        WaitForQueriesToFinishOrSaveState();
        m.Result = IntPtr.Zero;
        return;
    }
    base.WndProc(ref m);
}

关于c# - 如何取消PC的关机事件?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17389162/

相关文章:

c# - 如何在 C# 中将事件作为参数传递

c# - [,]在c#中是什么意思?

c# - TypeScript 数组的别名

c# - 将控件值从表单传递到单独类的最佳方法

C# .NET (WinForm) - MainForm 分为菜单(左)和内容(右,子窗体)

c# - 找不到源 'x' 中事件 ID 'y' 的描述。从 eventvwr C# 读取 'System' 事件日志

c# - 为什么 .NET System.IO.File 使用 Create/Open 而不是构造函数?

c# sql插入有很多参数

c# - 我什么时候应该注销系统事件以防止内存泄漏?

C# 创建其他应用程序可以看到的系统事件