6

我有一个表单,我FormClosing根据CloseReason. 例如,当从 WCF 服务调用返回特定值并从通知图标上下文菜单事件调用以在其他一些条件下关闭应用程序时,CloseReason.UserClosing我正在以编程方式调用事件时最小化表单。Application.Exit()ItemClicked

虽然到目前为止我的应用程序按预期工作,总是在被调用FormClosing时引发事件Application.Exit(),但在我进行一些更改后它的行为并不像那样。大多数情况下,应用程序在没有引发事件的情况下关闭,并且在少数情况下,事件的代码在不退出表单的情况下执行(数据库工作,通知图标处理)。

这怎么可能发生?

4

2 回答 2

14

是的,这是可能的。Application.Exit() 方法迭代 Application.OpenForms 集合中的表单以调用其 OnFormClosing() 方法。Winforms 中存在一个错误,导致该集合无法跟踪打开的表单。这段代码演示了它:

public partial class Form1 : Form {
    public Form1() {
        InitializeComponent();
    }
    protected override void OnHandleCreated(EventArgs e) {
        // Set breakpoint here:
        base.OnHandleCreated(e);
    }
    protected override void OnMouseDown(MouseEventArgs e) {
        this.ShowInTaskbar = !this.ShowInTaskbar;
        MessageBox.Show(string.Format("There are {0} open forms", Application.OpenForms.Count));
        Application.Exit();
    }
    protected override void OnFormClosing(FormClosingEventArgs e) {
        MessageBox.Show("you won't see this");
        base.OnFormClosing(e);
    }
}

单击表单以触发错误。请注意它如何报告 0 个打开的表单,以及您如何从未看到 OnFormClosing 中显示的消息框。

它是对 ShowInTaskbar 属性的分配导致它。有几个这样的属性,我选择 ShowInTaskbar 是因为当你有一个 NotifyIcon 时你可能会使用它。RightToLeft 是另一个。这些属性是特殊的,因为它们只能在使用本机 CreateWindowEx() api 函数创建窗口时指定。更改它们需要 Winforms 做一些非常英勇的事情,它会破坏窗口并重新创建它。不幸的是,这也会触发错误,破坏窗口也会从 OpenForms 集合中删除表单,并且忘记将其添加回来。

将此代码段中的 OnHandleCreated() 方法复制/粘贴到您的表单中并在其上设置断点。它必须在第一次创建窗口时触发一次。当它再次触发并因此调用错误场景时,您可以查看调用堆栈以查看类中的哪些代码触发了它。您必须禁用该代码并找到另一种方法来执行此操作。在构造函数中设置 ShowInTaskbar 属性很好,只有在创建窗口后分配它时才会变坏,就像在 Load 事件处理程序中一样。

于 2012-11-23T10:35:10.860 回答
1

正如这里所说

当调用 Application.Exit 方法退出应用程序时,不会引发 Form.Closed 和 Form.Closing 事件。如果您在其中任何一个事件中都有必须执行的验证代码,则应在调用 Exit 方法之前为每个打开的表单单独调用 Form.Close 方法。

编辑

由于您正在使用FormClosing,在这种情况下,您可能应该尝试使用其重载Exit可能有助于确定取消退出的表单并帮助您进一步检查问题。

于 2012-11-23T09:58:37.907 回答