在上节中说到,在win窗体之外的线程不可以直接对其引用进行访问,可通过Invoke()等几个方法和属性使用委托来实现。
例子很简单,在此就不截图了,窗体上只有一按钮一Lable,点击按钮开启一个线程来处理,在处理过程中Lable来显示处理进度。
此例子是通过Thread类来创建线程,按钮事件很简单,代码如下:
1
private void button1_Click(object sender, EventArgs e)
2
{
3
Thread thread = new Thread(new ThreadStart(ThreadRun));
4
thread.Start();
5
}
![](https://www.cnblogs.com/Images/OutliningIndicators/None.gif)
2
![](https://www.cnblogs.com/Images/OutliningIndicators/ExpandedBlockStart.gif)
3
![](https://www.cnblogs.com/Images/OutliningIndicators/InBlock.gif)
4
![](https://www.cnblogs.com/Images/OutliningIndicators/InBlock.gif)
5
![](https://www.cnblogs.com/Images/OutliningIndicators/ExpandedBlockEnd.gif)
在此实现化了一个新线程thread,并使用了无参数的ThreadStart,接着调用线程的Start()方法来开启线程。在此主要看下线程处理过程。
上面提到要使用Invoke方法来实现,此方法要用到的一个重载样式如下:
它接收一个委托类型以及一个object数组类型。下面就先定义一个委托,如下:
delegate void SetLableTextCallBack(string txt);
如上,此委托为一个带string参数的方法定义。再来实现此结构的方法,如下:
1
public void SetLableText(string text)
2
{
3
if (this.label1.InvokeRequired)
4
{
5
SetLableTextCallBack callBack = new SetLableTextCallBack(SetLableText);
6
this.Invoke(
7
callBack, new object[] { text });
8
}
9
else
10
{
11
this.label1.Text = text;
12
}
13
}
![](https://www.cnblogs.com/Images/OutliningIndicators/None.gif)
2
![](https://www.cnblogs.com/Images/OutliningIndicators/ExpandedBlockStart.gif)
3
![](https://www.cnblogs.com/Images/OutliningIndicators/InBlock.gif)
4
![](https://www.cnblogs.com/Images/OutliningIndicators/ExpandedSubBlockStart.gif)
5
![](https://www.cnblogs.com/Images/OutliningIndicators/InBlock.gif)
6
![](https://www.cnblogs.com/Images/OutliningIndicators/InBlock.gif)
7
![](https://www.cnblogs.com/Images/OutliningIndicators/ExpandedSubBlockStart.gif)
8
![](https://www.cnblogs.com/Images/OutliningIndicators/ExpandedSubBlockEnd.gif)
9
![](https://www.cnblogs.com/Images/OutliningIndicators/InBlock.gif)
10
![](https://www.cnblogs.com/Images/OutliningIndicators/ExpandedSubBlockStart.gif)
11
![](https://www.cnblogs.com/Images/OutliningIndicators/InBlock.gif)
12
![](https://www.cnblogs.com/Images/OutliningIndicators/ExpandedSubBlockEnd.gif)
13
![](https://www.cnblogs.com/Images/OutliningIndicators/ExpandedBlockEnd.gif)
如上,方法先对属性InvokeRequired进行判断,创建一个委托,并调用Invoke方法。
以上只是一个简单的例子,并没有实现对线程的其他(中断,取消,恢复)操作。
记得在页面关闭时一定要对线程进行终止,线程默认为前台线程,所以不会随着窗体的关闭而关闭,返而在线程处理窗体控件时会出现空引用等异常。
Over!