zoukankan      html  css  js  c++  java
  • C# 窗体Show和ShowDialog 方法的区别

    转自蓝色闪电博客 http://bluelightning.blog.163.com/blog/static/58165320101272172456/

    CenterParent                     窗体在其父窗体中居中。     
      CenterScreen                    窗体在当前显示窗口中居中,其尺寸在窗体大小中指定。     
      Manual                               窗体的位置由   Location   属性确定。     
      WindowsDefaultBounds     窗体定位在   Windows   默认位置,其边界也由   Windows   默认决定。     
      WindowsDefaultLocation    窗体定位在   Windows   默认位置,其尺寸在窗体大小中指定。     
        
      也就是说,CenterScreen的意思并不是屏幕居中(是相对的),它是在"当前显示窗口"中居中,当用Show()方法时应选择CenterScreen,用ShowDialog()方法时应选择CenterParent,这样才能让要显示的窗口居中。

      在SDI中用ShowDialog()方法,并且设置对应的窗体的StartPosition为CenterParent时就可以让窗体居中,当然也可以用CenterScreen也是一样的效果,只是含意不一样罢了.   
      而在MDI中只能用Show(),如果你用ShowDialog(),无论选择CenterParent或是CenterScreen都会出错,说ShowDialog只能用在顶级窗口之类的意思。用Show(),并且设置对应的窗体的StartPosition为CenterScreen时就可以让窗体居中。


    ShowDialog()    弹出模式化的窗体

    Show()              弹出非模式化的窗体

    模式窗体,在关闭或隐藏前无法切换到主窗体。

    非模式窗体,变换焦点使不必关闭窗体

    总结:显示重要的信息,还是用模式窗体,如删除文件,可以确保用户正真想要删除的是该文件

    非模式的,窗体访问的顺序没有办法得知,比较适合显示程序的一些相关信息。

    Application.Run()是"Begins running a standard application message loop on the current thread, and makes the specified form visible." 用代码可以表示为:
    while(GetMessage(&msg)>0)
    {
      TranslateMessage(&msg);
      DispatchMessage(&msg);
    }
    执导受到WM_QUIT,退出应用程序。而使用Form.Show()方法则Form显示后就立刻退出程序。如果用Form.ShowDialog()是模化对话框所以不会立刻消失,但是如果您还有其他窗口,Form.ShowDialog()显示的是模态窗口,只有它退出,其他窗口才能显示,但是一旦退出整个程序就退出,其他窗口将永不会得到运行机会,而用Application.Run()就不会出现这种现象。

    窗体显示中form1.Show()和form1.ShowDialog()的区别

    窗体和对话框要么是有模式的,要么是无模式的。“模式”窗体或对话框必须关闭或隐藏,然后您才能继续使用应用程序的其余部分。

    显示重要消息的对话框应始终是有模式的。模式对话框的一个示例是 Visual Studio 中的“关于”对话框。MessageBox是您可以使用的一个模式窗体。有模式对话框显示出来以后,那么开始打开的窗体或对话框就不能再获得焦点了。

    “无模式”窗体让您在此窗体与另一窗体之间变换焦点,而不必关闭初始窗体。用户在该窗体显示的同时可继续在任何应用程序的其他位置工作。例如:文本编辑软件里面的搜索功能,就是一个无模式的,因为搜索对话框出来以后,还可以操作编辑本文,即它不影响其他窗体获得焦点。

    将窗体显示为有模式对话框用form1.ShowDialog()方法。这个方法有一个可选参数 owner,该参数可用于指定窗体的父子关系。例如:

    在Form1代码段中:

    Form2 f2=new Form2();

    f2.ShowDialog(this);//this表示Form1当前实例

    这样f2实例就和Form1实例建立了一个父子关系,可以相互通讯。

    如果没有使用f2.ShowDialog(this)而直接使用的是无参的,要定义父子关系,则需要语句f2.owner=this;

    将窗体显示为无模式对话框则用form1.Show()方法。

    注意   如果窗体显示为有模式,则在关闭该对话框之前,不执行 ShowDialog 方法后面的代码。但是,当窗体显示为无模式时,那么该窗体显示之后,会立刻执行 Show 方法后面的代码。

    窗体的Show方法,没有给调用代码任何通知,如果需要通知,使用ShowDialog是一种好的选择。 
    在调用Show方法后,Show方法后面的代码会立即执行,调用ShowDialog方法后,调用代码被暂停执行,等到调用ShowDialog方法的窗体关系后再继续执行。而且窗体可以返回一个dialogresult值,他描述了窗体关闭的原因,例如OK,Cancel,yes,no等。为了让窗体返回一个dialogresult,必须设置窗体的dialogresult值,或者在窗体的一个按钮上设置dialogresult属性。

    例子:
    下面是子窗体代码,要求输入phone,然后会返回给父窗体。
    using System;
    using System.Collections.Generic;
    using System.ComponentModel;
    using System.Data;
    using System.Drawing;
    using System.Text;
    using System.Windows.Forms;
    namespace WindowsApplication1
    {
      public partial class Phone : Form
      {
        public Phone()
        {
          InitializeComponent();
          btnOK.DialogResult = DialogResult.OK;
          btnOK.DialogResult = DialogResult.Cancel;
        }
        public string PhoneNumber
        {
          get { return textBox1.Text; }
          set { textBox1.Text = value; }
        }
        private void Phone_Load(object sender, EventArgs e)
        {
        }
      }
    }
      不包含任何处理按钮单击事件的代码,因为设置了每个按钮的dialogresult属性,所以单击OK或者Cancel按钮后,窗体就消失了。下面的代码显示了父窗体中调用Phone对话框的方法。


    Form.Show和Form.ShowDialog的区别
    区别1:ShowDialog是模态的(独占用户输入),Show是非模态的。
    区别2:根据1,ShowDialog只能打开一个自己,Show可以打开多个自己。
    区别3:根据2,使用Show方法打开的Form在关闭时会立即调用Dispose释放资源。那ShowDialog会在关闭时立即释放资源吗?我做了个实验。

    public partial class Form1 : Form
      {
          private Form2 f2 = new Form2();

          public Form1()
          {
              InitializeComponent();
          }

          private void button1_Click(object sender, EventArgs e)
          {
            try
            {
                f2.Show();
            }
            catch (Exception ex)
            {
                MessageBox.Show(ex.Message);
            }
          }

          private void button2_Click(object sender, EventArgs e)
          {
            try
            {
                f2.ShowDialog();
            }
            catch (Exception ex)
            {
                MessageBox.Show(ex.Message);
            }
          }

          private void button3_Click(object sender, EventArgs e)
          {
              f2 = new Form2();
          }
      }
      
          
    连续点击button1会提示“无法访问已释放的对象。对象名:Form2”。而连续点击button2则不然,一切正常。继续调查发现,使用ShowDialog时,第一次会调用构造函数Form2(),然后依次调用Form2_Load,Form2_Activated,Form2_Shown。第二次及以后使用ShowDialog时,只会调用Form2_Load,Form2_Activated,Form2_Shown。由此可以判断,Form2在关闭时只是被隐藏,而非解构。

          ShowDialog的这种设计也是有道理的,在连续调用时可以节省资源,但也要警惕"上辈子"影响到"这辈子"。如果想及时释放资源,在调用Form.ShowDialog()后面不妨加一句Form.Close(),Close方法会调用Dispose解构Form。

  • 相关阅读:
    Vue.js 2.x笔记:安装与起步(1)
    EntityFramework Core笔记:保存数据(4)
    EntityFramework Core笔记:查询数据(3)
    EntityFramework Core笔记:表结构及数据基本操作(2)
    EntityFramework Core笔记:入门(1)
    ASP.NET MVC系列:web.config中ConnectionString aspnet_iis加密与AppSettings独立文件
    EntityFramework优化:第一次启动优化
    EntityFramework优化:查询性能
    EntityFramework优化:查询WITH(NOLOCK)
    SpringCloud学习笔记:熔断器Hystrix(5)
  • 原文地址:https://www.cnblogs.com/12xiaole/p/7508419.html
Copyright © 2011-2022 走看看