C#开发Winform实现窗体间相互传值

 更新时间:2022年03月20日 15:43:52   作者:.NET开发菜鸟  
这篇文章介绍了C#开发Winform实现窗体间相互传值的方法,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

一、前言

我们在做Winform窗体程序开发的时候,会经常遇到窗体之间相互传值。假设有下面的一个场景:一个主窗体和一个子窗体,点击主窗体上面的按钮给子窗体传值,并在子窗体上面显示出来,一般会有如下几种方式实现。

二、公共属性

我们可以在子窗体里面定义一个公共的属性,然后在父窗体里面给公共属性赋值,这样可以实现窗体之间传值,子窗体代码如下:

using System;
using System.Windows.Forms;

namespace DelegateDemo
{
    public partial class frmChild : Form
    {
        public frmChild()
        {
            InitializeComponent();
        }

        // 定义一个公共属性,接收传递的值
        public string strMessage { get; set; }

        /// <summary>
        /// 窗体加载
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void frmChild_Load(object sender, EventArgs e)
        {
            // 将接收到的值显示在窗体上
            this.lblMessage.Text = strMessage;
        }
    }
}

父窗体代码:

using System;
using System.Windows.Forms;

namespace DelegateDemo
{
    public partial class frmParent : Form
    {
        public frmParent()
        {
            InitializeComponent();
        }

        /// <summary>
        /// 单击事件
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void btn_Value_Click(object sender, EventArgs e)
        {
            frmChild child = new frmChild();
            // 给窗体的公共属性赋值
            child.strMessage = this.txtMessage.Text.Trim();
            // 显示子窗体
            child.Show();
        }
    }
}

这种方式有一个缺点:属性需要设置为public,不安全。

三、公共方法

我们还可以在子窗体里面定义一个方法,通过调用方法传值,子窗体代码如下:

using System;
using System.Windows.Forms;

namespace DelegateDemo
{
    public partial class frmChild : Form
    {
        public frmChild()
        {
            InitializeComponent();
        }

        // 定义一个公共属性,接收传递的值
        //public string strMessage { get; set; }

        // 定义属性为private
        private string strMessage { get; set; }

        /// <summary>
        /// 给私有属性赋值
        /// </summary>
        /// <param name="strText"></param>
        public void SetText(string strText)
        {
            strMessage = strText;
        }

        /// <summary>
        /// 窗体加载
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void frmChild_Load(object sender, EventArgs e)
        {
            // 将接收到的值显示在窗体上
            this.lblMessage.Text = strMessage;
        }
    }
}

父窗体代码:

using System;
using System.Windows.Forms;

namespace DelegateDemo
{
    public partial class frmParent : Form
    {
        public frmParent()
        {
            InitializeComponent();
        }

        /// <summary>
        /// 单击事件
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void btn_Value_Click(object sender, EventArgs e)
        {
            #region 调用公共属性赋值
            //frmChild child = new frmChild();
            //// 给窗体的公共属性赋值
            //child.strMessage = this.txtMessage.Text.Trim();
            //// 显示子窗体
            //child.Show(); 
            #endregion


            #region 调用方法赋值
            frmChild child = new frmChild();
            // 给窗体的公共属性赋值
            child.SetText(this.txtMessage.Text.Trim());
            // 显示子窗体
            child.Show(); 
            #endregion
        }
    }
}

这种方式同样也有缺点:属性虽然是private的了,但是方法还是public的。

四、委托

上述两种方式都是不安全,下面我们使用委托来实现窗体之间传值。

1、定义一个委托

我们在主窗体里面定义一个有参无返回值的委托:

// 定义一个有参无返回值的委托
private delegate void SendMessage(string strMessage);

2、实例化一个此委托类型的事件

在父窗体里面定义一个委托类型的事件:

// 定义一个委托类型的事件
public event SendMessage sendMessageEvent;

委托与事件的关系,事件相对于委托更安全,更低耦合。委托是一个类型,事件是委托类型的一个实例。

3、定义要执行的方法

这里其实就是在子窗体里面定义一个给控件赋值的方法:

/// <summary>
/// 给控件赋值的方法
/// </summary>
/// <param name="strValue"></param>
public void SetValue(string strValue)
{
    this.lblMessage.Text = strValue;
}

4、将方法绑定到事件

frmChild child = new frmChild();
// 将方法绑定到事件上
sendMessageEvent += new SendMessage(child.SetValue);
// 也可以使用下面的简写形式
// sendMessageEvent += child.SetValue;
child.Show();

5、触发委托

在按钮的点击事件里面触发委托:

if(sendMessageEvent!=null)
{
      sendMessageEvent.Invoke(this.txtMessage.Text.Trim());
}

上面的代码中使用的是自定义的委托,我们也可以使用.Net 框架里面自带的Action泛型委托:

using System;
using System.Windows.Forms;

namespace DelegateDemo
{
    
    public partial class frmParent : Form
    {
        // 定义一个有参无返回值的委托
        public delegate void SendMessage(string strMessage);

        // 定义一个委托类型的事件
        public event SendMessage sendMessageEvent;


        public event Action<string> actionEvent;
        public frmParent()
        {
            InitializeComponent();
        }

        /// <summary>
        /// 单击事件
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void btn_Value_Click(object sender, EventArgs e)
        {
            #region 调用公共属性赋值
            //frmChild child = new frmChild();
            //// 给窗体的公共属性赋值
            //child.strMessage = this.txtMessage.Text.Trim();
            //// 显示子窗体
            //child.Show(); 
            #endregion


            #region 调用方法赋值
            //frmChild child = new frmChild();
            //// 给窗体的公共属性赋值
            //child.SetText(this.txtMessage.Text.Trim());
            //// 显示子窗体
            //child.Show(); 
            #endregion

            #region 通过委托传值
            //frmChild child = new frmChild();
            //// 将方法绑定到事件上
            //// sendMessageEvent += new SendMessage(child.SetValue);
            //// 也可以使用下面的简写形式
            //sendMessageEvent += child.SetValue;
            //child.Show();
            #endregion

            #region 使用Action
            frmChild child = new frmChild();
            // 将方法绑定到事件上
            actionEvent += child.SetValue;
            child.Show();
            #endregion

            // 使用自定义委托
            //if (sendMessageEvent!=null)
            //{
            //    sendMessageEvent.Invoke(this.txtMessage.Text.Trim());
            //}

            // 使用Action委托
            if (actionEvent != null)
            {
                actionEvent.Invoke(this.txtMessage.Text.Trim());
            }
        }
    }
}

完整示例代码:https://github.com/jxl1024/FromPassValueDemo

到此这篇关于C#开发Winform实现窗体间相互传值的文章就介绍到这了。希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • C# WebClient类用法实例

    C# WebClient类用法实例

    这篇文章主要介绍了C# WebClient类用法实例,本文讲解使用WebClient下载文件、OpenWriter打开一个流使用指定的方法将数据写入到uri以及上传文件示例,需要的朋友可以参考下
    2015-07-07
  • 详解C# 不能用于文件名的字符

    详解C# 不能用于文件名的字符

    在 Windows 有一些字符是不能作为文件名,尝试重命名一个文件,输入/ 就可以看到windows 提示的不能作为文件名的字符,那么具体是包括哪些符号不能作为文件名呢?下面小编给大家介绍下
    2018-02-02
  • 详解Unity入门之GameObject

    详解Unity入门之GameObject

    Unity是一个Component-Based的引擎,所有物体都是GameObject。本文将详细介绍GameObject和MonoBehaviour,感兴趣的同学,可以参考下。
    2021-05-05
  • C#常用目录文件操作类实例

    C#常用目录文件操作类实例

    这篇文章主要介绍了C#常用目录文件操作类,实例分析了C#针对目录的读取、检测及查找等相关操作技巧,非常具有实用价值,需要的朋友可以参考下
    2015-03-03
  • C#数值转换-显式数值转换表(参考)

    C#数值转换-显式数值转换表(参考)

    就是在将一种类型转换成另外一种类型时,需要额外的代码来完成这种转换。
    2013-04-04
  • C#任务并行Parellel.For和Parallel.ForEach

    C#任务并行Parellel.For和Parallel.ForEach

    这篇文章介绍了C#任务并行Parellel.For和Parallel.ForEach的用法,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-07-07
  • C#中Kestrel和IIS服务器下的同步与异步配置

    C#中Kestrel和IIS服务器下的同步与异步配置

    本篇文章主要讲解什么是Kestrel和IIS服务器和特点,以及他们如何配置同步与异步,具有一定的参加价值,感兴趣的可以了解一下
    2023-08-08
  • Unity3D实现人物转向与移动

    Unity3D实现人物转向与移动

    这篇文章主要为大家详细介绍了Unity3D实现人物转向与移动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-01-01
  • C#泛型设计需要注意的一个小陷阱

    C#泛型设计需要注意的一个小陷阱

    这篇文章主要给大家介绍了关于C#泛型设计需要注意的一个小陷阱,文中通过示例代码以及图文介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-03-03
  • C# SQLite库使用技巧

    C# SQLite库使用技巧

    SQLite是一个开源、免费的小型RDBMS(关系型数据库),能独立运行、无服务器、零配置、支持事物,用C实现,内存占用较小,支持绝大数的SQL92标准。下面跟随小编一起看下C# SQLite库使用
    2022-01-01

最新评论