我的操作系统是Win7旗舰版,VS版本是VS2012,.NET版本为.NET Framework 4.5。
在窗体的FormClosing事件,第二个参数(FormClosingEventArgs类型)下有一个枚举变量CloseReason,在窗体的FormClosed事件,第二个参数(FormClosedEventArgs类型)下也有一个枚举变量CloseReason 。这个CloseReason枚举在命名空间System.Windows.Forms下。
如下段代码所示,CloseReason在窗体FormClosing事件的FormClosingEventArgs类型变量e中。
private void FormMain_FormClosing(object sender, FormClosingEventArgs e)
{
switch (e.CloseReason)
{
case CloseReason.None: { MessageBox.Show("Closing: CloseReason.None"); } break;
case CloseReason.WindowsShutDown: { MessageBox.Show("Closing: CloseReason.WindowsShutDown"); } break;
case CloseReason.MdiFormClosing: { MessageBox.Show("Closing: CloseReason.MdiFormClosing"); } break;
case CloseReason.UserClosing: { MessageBox.Show("Closing: CloseReason.UserClosing"); } break;
case CloseReason.TaskManagerClosing: { MessageBox.Show("Closing: CloseReason.TaskManagerClosing"); } break;
case CloseReason.FormOwnerClosing: { MessageBox.Show("Closing: CloseReason.FormOwnerClosing"); } break;
case CloseReason.ApplicationExitCall: { MessageBox.Show("Closing: CloseReason.ApplicationExitCall"); } break;
}
}
从元数据看,该枚举一共有下面7个枚举值:
#region 程序集 System.Windows.Forms.dll, v4.0.0.0
// C:\Program Files\Reference Assemblies\Microsoft\Framework\.NETFramework\v4.0\System.Windows.Forms.dll
#endregion
using System;
namespace System.Windows.Forms
{
// 摘要:
// 指定窗体关闭的原因。
public enum CloseReason
{
// 摘要:
// 关闭原因未定义或者无法确定。
None = 0,
//
// 摘要:
// 操作系统正在关闭所有应用程序以便准备关机。
WindowsShutDown = 1,
//
// 摘要:
// 此多文档界面 (MDI) 窗体的父窗体正在关闭。
MdiFormClosing = 2,
//
// 摘要:
// 用户正在通过用户界面 (UI) 关闭该窗体,例如通过单击窗体窗口上的“关闭”按钮,通过选择窗口控制菜单上的“关闭”按钮,或者通过按 Alt+F4
// 等方式关闭。
UserClosing = 3,
//
// 摘要:
// Microsoft Windows 任务管理器正在关闭应用程序。
TaskManagerClosing = 4,
//
// 摘要:
// 所有者窗体正在关闭。
FormOwnerClosing = 5,
//
// 摘要:
// System.Windows.Forms.Application 类的 System.Windows.Forms.Application.Exit()
// 方法被调用。
ApplicationExitCall = 6,
}
}
该枚举的MSDN可以参考页面:
https://msdn.microsoft.com/en-us/library/system.windows.forms.closereason(v=vs.110).aspx
除去None类型,本文对6种枚举值都进行了测试,在此记录下测试的结果。
1、CloseReason.WindowsShutDown
当Windows被注销、关闭时触发此CloseReason,不过不要在这里加入MessageBox等元素,因为一旦Windows发现当前程序迟迟关不掉,就会将此程序强制关闭。
2、CloseReason.MdiFormClosing
当前窗体为Mdi子窗体时,Mdi容器窗体被关闭时,在触发当前的FormClosing和FormClosed事件时提示此CloseReason。
将当前窗体作为MdiParent打开另一窗体的方法:
FormChild formChild = new FormChild();
formChild.MdiParent = this;
formChild.Show();
(需要将本窗体的IsMdiContainer设置为True)
3、CloseReason.UserClosing
用户手动关闭当前程序,比如调用Close()函数,或点击程序右上角的“×”,关闭原因都是CloseReason.UserClosing。
4、CloseReason.TaskManagerClosing
由任务管理器关闭窗口时,会触发此事件,但我在测试的时候发现,任务管理器关闭窗口具有一定的强制性。设置断点后可发现,FormClosing事件触发后不久程序就会被任务管理器强制关闭,这个时间非常短,因此不适合在此做一些诸如弹出MessageBox的事情(因为没有用)。
5、CloseReason.FormOwnerClosing
类似CloseReason.MdiFormClosing,如果窗体A是窗体B的owner,则窗体A关闭时,窗体B触发FormClosing和FormClosed事件时使用此CloseReason。
关于窗体作为owner的问题,可以参考MSDN页面:
https://msdn.microsoft.com/en-us/library/system.windows.window.owner(v=vs.110).aspx
将当前窗体作为Owner打开另一个窗体的方法:
FormChild formChild2 = new FormChild();
formChild2.Owner = this;
formChild2.Show();
6、CloseReason.ApplicationExitCall
调用Application.Exit()方法退出程序时,CloseReason为此值。
最后再说下FormClosing、FormClosed事件的调用顺序:
1、FormClosing事件在窗体关闭前触发,FormClosed事件在窗体关闭后触发
2、如果窗体A是mdi容器,窗体B的mdi-parent是窗体A,那么事件的调用顺序是:
窗体B - FormClosing事件 - CloseReason.MdiFormClosing
窗体A - FormClosing事件 - CloseReason.UserClosing
窗体B - FormClosed事件 - CloseReason.MdiFormClosing
窗体A - FormClosed事件 - CloseReason.UserClosing
3、如果窗体A是窗体B的owner,那么事件的调用顺序是:
窗体B - FormClosing事件 - CloseReason.FormOwnerClosing
窗体A - FormClosing事件 - CloseReason.UserClosing
窗体B - FormClosed事件 - CloseReason.FormOwnerClosing
窗体A - FormClosed事件 - CloseReason.UserClosing
最后分享下我用于测试此问题的C#工程,该工程使用VS2012建立:
http://pan.baidu.com/s/1pLF3FJH
END