C#多线程访问主窗体控件

先看这样的一个例子:

点击"多线程访问"按钮标签中文本"此标签被另一个线程设置文本"会变为"Hello"!
代码是这样写的:
/// <summary>
///
 设置标签的文本
/// </summary>
private void SetLableText()
{
     this.label1.Text = "Hello!";
}
 /// <summary>


///
 设置标签的按钮事件
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void button1_Click(object sender, EventArgs e)
{
     System.Threading.Thread setLabelTextThread = new System.Threading.Thread(new System.Threading.ThreadStart(this.SetLableText));
     setLabelTextThread.Start();    
}
按照想法,这个功能是完成了,运行.点击按钮,却出现了异常:

分析:label标签控件是主线程创建的,不能直接从另一个线程访问.可以这样认为:不能跨线程直接访问控件;
如何才能实现这个功能呢?
.NET,所有的控件都是从System.Windows.Forms.Control类派生,Control类提供了一个Invoke()方法,用于在创建控件的线程中访问线程.它的定义如下:
public Object Invoke(Delegate method);
它的参数为一个委托,代表创建控件的线程中要执行的方法.
可以利用这个方法来实现这个功能.
首先定义一个委托:
public delegate void setLabelTextDelegate();//
定义一个setLabelTextDelegate()的委托
在定义一个委托变量:
private setLabelTextDelegate setLabelText;
在窗体的构造函数中给这个委托变量初始化:
public Form1()
 {
       InitializeComponent();
       this.setLabelText = this.SetLableText;//SetLableText
为上面的"设置标签的文本"的方法
 }

 

小白注:可以把委托看成是函数指针,注意setLabelText和SetLableText是区分大小写的,前者是委托变量,后者是原操作函数。


然后在定义一个方法.方法里使用Invoke
private void ThreadMethod()
 {
       this.label1.Invoke(this.setLabelText);//setLabelText
为上面定义的委托变量
 }
接着把按钮事件里的代码修改一下:
/// <summary>
        ///
 设置标签的按钮事件
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void button1_Click(object sender, EventArgs e)
        {
            System.Threading.Thread setLabelTextThread = new System.Threading.Thread(new System.Threading.ThreadStart(
this.SetLableText));//这个方法修改为ThreadMethod,:
// System.Threading.Thread setLabelTextThread = new System.Threading.Thread(new System.Threading.ThreadStart(this.ThreadMethod));

            setLabelTextThread.Start();
        }
这个就OK了,运行.点击:

功能实现:

不过 " 设置标签的文本 " 的方法 SetLableText() 是没有参数的,在很多情况下 , 我们写的方法都是需要参数的 , 下面我就把这个例子改成有参数的 , 并演示如何传递参数 :
首先 : 改造 " 设置标签的文本 " 的方法 SetLableText() 变成有参数的 :
/// <summary>
///
  设置标签的文本
/// </summary>
private void SetLableText(string info)
{
     this.label1.Text = info;
 }
既然这个方法有参数了,与它对应的委托应该使用参数 :
public delegate void setLabelTextDelegate(string infor);

定义的委托变量还是在构造函数中初始化 , 这个不用改变什么 :
public Form1()
 {
       InitializeComponent();
       this.setLabelText = this.SetLableText;
 }
既然使用了参数 , 那么 Invoke() 这个方法应该会有重载的方法吧 ?
Invoke() 这个方法是有重载的 , 它的定义如下 :
public Object Invoke(Delegate method,param Object [ ] args);
第二个参数是一个 object 的数组 , 就意味着 , 可以把需要传递的参数放到这个数组里面来进行传递
ThreadMethod( ) 改造 :
private void ThreadMethod( Object info )
 {
        this.label1.Invoke(this.setLabelText,
  new object[] { info});
 }
注意红色部分,为添加的参数
最后是按钮事件的改造了 :

/// <summary>
        ///
  设置标签的按钮事件
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void button1_Click(object sender, EventArgs e)
        {
           
  System.Threading.Thread setLabelTextThread = new System.Threading.Thread(this.ThreadMethod);
            setLabelTextThread.Start("Hello!");
           
        }

改造成功 :

跨线程访问控件步骤可以总结一下 :
(1)
将访问的控件代码封装为一个方法 ;
(2)
根据方法自定义一个对应委托 ;
(3)
增加一个定义的委托类型的字段 , 并把前面访问控件的方法 " 挂接 " 到此字段中 ;
(4)
编写一个线程方法 , 在此方法中调用要访问控件的 Invoke 方法 , 并把定义好了的委托字段做为参数传入 .
(5)
在合适的地方创建线程并启动运行

猜你喜欢

转载自blog.csdn.net/jx_521/article/details/11259653