先看这样的一个例子:
点击"多线程访问"按钮标签中文本"此标签被另一个线程设置文本"会变为"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() 变成有参数的 :
/// <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) 在合适的地方创建线程并启动运行