是的,这是可能的。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 事件处理程序中一样。