关于Form.Close跟Form.Dispose

 

我们在Winform开发的时候,使用From.Show来显示窗口,使用Form.Close来关闭窗口。熟悉Winform开发的想必对这些非常熟悉。但是Form类型实现了IDisposable接口,那我们是否需要每次关闭窗口后都去调用Dispose呢?对于这个问题我们可以查看一下Form的源码。

Form.Close
 public void Close()
{
if (this.GetState(262144))
throw new InvalidOperationException(SR.GetString("ClosingWhileCreatingHandle", new object[1]
{
(object) "Close"
}));
else if (this.IsHandleCreated)
{
this.closeReason = CloseReason.UserClosing;
this.SendMessage(16, 0, 0);
}
else
base.Dispose();
}
很明显这个方法有3个分支。第一个分支是关闭出现异常的情况,第二个分支是句柄已经创建的时候执行,很明显第三个分支的时候直接调用了基类的Dispose方法。大部分时候窗口调用Close时句柄肯定是被创建了,那就会进入第二个分支。很明显第二个分支没有调用Dispose,那是不是真的呢?继续跟进去。
 
SendMessage  
internal IntPtr SendMessage(int msg, int wparam, int lparam)
{
return UnsafeNativeMethods.SendMessage(new HandleRef((object) this, this.Handle), msg, wparam, lparam);
}
调了一个静态方法继续进去
    [DllImport("user32.dll", CharSet = CharSet.Auto)]
public static IntPtr SendMessage(HandleRef hWnd, int msg, int wParam, int lParam);

明白了吧。SendMessage其实是调用了user32.dll的API,向指定的窗口发送消息。这里就是向自己发送一个消息,注意msg=16哦。既然有发送消息的,那肯定得有拦截消息的方法啊。(这里多扯一句,.NET Winform使用了事件驱动机制,事件机制其实也是封装了消息机制。)

WndProc
    protected override void WndProc(ref Message m)
{
switch (m.Msg)
{
case 529:
this.WmEnterMenuLoop(ref m);
break;
case 530:
this.WmExitMenuLoop(ref m);
break;
case 533:
base.WndProc(ref m);
if (!this.CaptureInternal || Control.MouseButtons != MouseButtons.None)
break;
this.CaptureInternal = false;
break;
case 546:
this.WmMdiActivate(ref m);
break;
case 561:
this.WmEnterSizeMove(ref m);
this.DefWndProc(ref m);
break;
case 562:
this.WmExitSizeMove(ref m);
this.DefWndProc(ref m);
break;
case 288:
this.WmMenuChar(ref m);
break;
case 293:
this.WmUnInitMenuPopup(ref m);
break;
case 274:
this.WmSysCommand(ref m);
break;
case 279:
this.WmInitMenuPopup(ref m);
break;
case 167:
case 171:
case 161:
case 164:
this.WmNcButtonDown(ref m);
break;
case 71:
this.WmWindowPosChanged(ref m);
break;
case 130:
this.WmNCDestroy(ref m);
break;
case 132:
this.WmNCHitTest(ref m);
break;
case 134:
if (this.IsRestrictedWindow)
this.BeginInvoke((Delegate) new MethodInvoker(this.RestrictedProcessNcActivate));
base.WndProc(ref m);
break;
case 16:
if (this.CloseReason == CloseReason.None)
this.CloseReason = CloseReason.TaskManagerClosing;
this.WmClose(ref m);

break;
case 17:
case 22:
this.CloseReason = CloseReason.WindowsShutDown;
this.WmClose(ref m);
break;
case 20:
this.WmEraseBkgnd(ref m);
break;
case 24:
this.WmShowWindow(ref m);
break;
case 36:
this.WmGetMinMaxInfo(ref m);
break;
case 1:
this.WmCreate(ref m);
break;
case 5:
this.WmSize(ref m);
break;
case 6:
this.WmActivate(ref m);
break;
default:
base.WndProc(ref m);
break;
}
}

WndProc就是用来拦截窗口消息的。看一下代码,Form重写了这个方法,一个很简单的switch。Case 16调用了 WmClose方法,继续跟进去。

WmClose
private void WmClose(ref Message m)
{
FormClosingEventArgs e1 = new FormClosingEventArgs(this.CloseReason, false);
if (m.Msg != 22)
{
if (this.Modal)
{
if (this.dialogResult == DialogResult.None)
this.dialogResult = DialogResult.Cancel;
this.CalledClosing = false;
e1.Cancel = !this.CheckCloseDialog(true);
}
else
{
e1.Cancel = !this.Validate(true);
if (this.IsMdiContainer)
{
FormClosingEventArgs e2 = new FormClosingEventArgs(CloseReason.MdiFormClosing, e1.Cancel);
foreach (Form form in this.MdiChildren)
{
if (form.IsHandleCreated)
{
form.OnClosing((CancelEventArgs) e2);
form.OnFormClosing(e2);
if (e2.Cancel)
{
e1.Cancel = true;
break;
}
}
}
}
Form[] ownedForms = this.OwnedForms;
for (int index = this.Properties.GetInteger(Form.PropOwnedFormsCount) - 1; index >= 0; --index)
{
FormClosingEventArgs e2 = new FormClosingEventArgs(CloseReason.FormOwnerClosing, e1.Cancel);
if (ownedForms[index] != null)
{
ownedForms[index].OnFormClosing(e2);
if (e2.Cancel)
{
e1.Cancel = true;
break;
}
}
}
this.OnClosing((CancelEventArgs) e1);
this.OnFormClosing(e1);
}
if (m.Msg == 17)
m.Result = (IntPtr) (e1.Cancel ? 0 : 1);
if (this.Modal)
return;

}
else
e1.Cancel = m.WParam == IntPtr.Zero;
if (m.Msg == 17 || e1.Cancel)
return;
this.IsClosing = true;
if (this.IsMdiContainer)
{
FormClosedEventArgs e2 = new FormClosedEventArgs(CloseReason.MdiFormClosing);
foreach (Form form in this.MdiChildren)
{
if (form.IsHandleCreated)
{
form.OnClosed((EventArgs) e2);
form.OnFormClosed(e2);
}
}
}
Form[] ownedForms1 = this.OwnedForms;
for (int index = this.Properties.GetInteger(Form.PropOwnedFormsCount) - 1; index >= 0; --index)
{
FormClosedEventArgs e2 = new FormClosedEventArgs(CloseReason.FormOwnerClosing);
if (ownedForms1[index] != null)
{
ownedForms1[index].OnClosed((EventArgs) e2);
ownedForms1[index].OnFormClosed(e2);
}
}
FormClosedEventArgs e3 = new FormClosedEventArgs(this.CloseReason);
this.OnClosed((EventArgs) e3);
this.OnFormClosed(e3);
base.Dispose();
}

WmClose这个方法略有点复杂,主要是用来出发OnCloseing,OnClosed等事件。看看最后,它终于调用了base.Dispose()。看来Close方法确实会自动调用Dispose。是吗,不要高兴的太早。仔细看看前面的代码,if (this.Modal) return; 看到没,当窗口是模态的时候,方法直接return了。

总结

到这里就差不多了。所以当我们使用ShowDialog来显示窗体的时候,当你关闭的时候,最好手动Dispose一下。为什么是最好呢,因为其实在GC回收垃圾的时候还是会调用窗体的Dispose的,因为在Form的基类的终结器里面有调用Dispose(false);

   ~Component()
{
this.Dispose(false);
}

其实在MSDN上微软就对这有说明,顺便吐槽一下中文MSDN的翻译,实在是太烂了。

有2种情况下需要手工调用Dispose:
1. 窗口是MDI的一部分且是不可见的

2.模态的时候
第二种情况就是现在说的,但是第一种情况我测试了下,没有复现出来,MDI里面的子窗口调用Close的时候跟正常一样,每次都会自动Dispose。试了很久还是一样的结果,求高人指定吧。


BTW:求 苏州,上海地区有激情,有意义的技术类工作!!

Email:kklldog@gmail.com 
作者:Agile.Zhou(kklldog) 
出处:http://www.cnblogs.com/kklldog/ 
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

 
 
标签: Winform

Form.Close跟Form.Dispose的更多相关文章

  1. 关于Form.Close跟Form.Dispose

    我们在Winform开发的时候,使用From.Show来显示窗口,使用Form.Close来关闭窗口.熟悉Winform开发的想必对这些非常熟悉.但是Form类型实现了IDisposable接口,那我 ...

  2. 使用Form Builder创建Form具体步骤

    使用Oracle Form Builder创建Form具体步骤 (Data Source为Table) 说明:当Block使用的Data Source为Table时,Form会自动Insert,Upd ...

  3. 知道Form.Show()和Form.ShowDialog()的区别吗

    博客搬到了fresky.github.io - Dawei XU,请各位看官挪步.最新的一篇是:知道Form.Show()和Form.ShowDialog()的区别吗.

  4. form.Show()和form.ShowDialog()的区别、新建一个form和MessageBox.Show()的常见用法

    一:form.Show()和form.ShowDialog()的区别 a. 任何窗体(派生于基类Form的类),都可以以两种方式进行显示. //非模式窗体From qform=new Form();q ...

  5. $(#form :input)与$(#form input)的区别

    相信大家都很奇怪这两者的区别 我从两个方面简单介绍下 1. $("form :input") 返回form中的所有表单对象,包括textarea.select.button等    ...

  6. day75 form 组件(对form表单进行输入值校验的一种方式)

    我们的组件是什么呢 select distinct(id,title,price) from book ORM: model.py class Book(): title=model.CharFiel ...

  7. {Django基础十之Form和ModelForm组件}一 Form介绍 二 Form常用字段和插件 三 From所有内置字段 四 字段校验 五 Hook钩子方法 六 进阶补充 七 ModelForm

    Django基础十之Form和ModelForm组件 本节目录 一 Form介绍 二 Form常用字段和插件 三 From所有内置字段 四 字段校验 五 Hook钩子方法 六 进阶补充 七 Model ...

  8. http://www.vaikan.com/docs/jquery.form.plugin/jquery.form.plugin.html#getting-started

    http://www.vaikan.com/docs/jquery.form.plugin/jquery.form.plugin.html#getting-started Jquery.Form 异步 ...

  9. ExtJs6解决添加和修改Form共用一个form的隐藏域的id的取消传值

    问题重现:修改不会有问题,id会绑定之前的grid,有具体数字 添加有问题,因为id是空,传的是绑定值的话会显示“类名-1”,从int类型变成了string类型,后台会出错 这是EduQuestion ...

随机推荐

  1. maven和libgdx

    这一篇是关于maven和libgdx的.本来我准备用gradle(现已有gradle模板了),不过暂时有点小问题,而同时libgdx官方提供了maven支持,为了快速上手还是选用maven了. 博客已 ...

  2. linux下一个Oracle11g RAC建立(八)

    linux下一个Oracle11g RAC建立(八) 七.安装oracle软件   直接在图形界面里安装oracle.在node1操作 在虚拟机界面中,直接切换到oracle用户下: [grid@no ...

  3. mono for android 学习记录

    C#开发Android应用实战(全 扫描 中文版) 学习记录: 拖完控件后,不要急着按F5,需要重新生成,才能自动修改 Resource.Designer.cs 文件 1. Activity 是基于a ...

  4. C++内存泄露的有效预防方法:谁使用,谁删除 (1.2)

    内存泄露就是new出来的东西没有delete,我们能够这样:创建动态对象的人虽然使用new来创建对象:使用此对象的人负责释放此内存块. 比如:我和他人共享一个消息队列,他人将消息(new出来的对象)放 ...

  5. 程序员面试必备经典CTCI,谷歌面试官经典作品!

    1.1 判断一个字符串中的字符是否唯一 1.2 字符串翻转 1.3 去除字符串中重复字符 1.8 利用已知函数判断字符串是否为另一字符串的子串 2.1 从链表中移除重复结点 2.2 实现一个算法从一个 ...

  6. automake/autoconf的简单例子

    参考文章1:http://loftor.com/archives/automake.html 参考文章2:http://www.blogjava.net/huyi2006/articles/18790 ...

  7. UVA10537 Toll! Revisited

    difkstra + 路径输出 The Toll! Revisited Time Limit: 3000MS Memory Limit: Unknown 64bit IO Format: %lld & ...

  8. 在Ubuntu下直接通过SSH登录到OpenWrt

    先前一直使用的是putty,这个工具会另外打开一个x-term来显示,界面较难看点. 刚刚测试使用Ubuntu自己的SSH,很简便,命令为:(root为主机名,10.0.11.233为主机地址) ss ...

  9. ventBroker简单实现

    C#编程实践—EventBroker简单实现 前言 话说EventBroker这玩意已经不是什么新鲜货了,记得第一次接触这玩意是在进第二家公司的时候,公司产品基础架构层中集成了分布式消息中间件,在.n ...

  10. Oracle的SOME,ANY和ALL操作

    平时很少用的这几个操作,今天遇到了.于是又看了一下文档. SOME和ANY一样,是比较宽松的,类似于OR.满足其中任何一个都可以. ALL要求严格一些,类似于AND,必须全部满足才可以. 不能单独使用 ...