开始正题前,大概叙述下委托,委托在.net 中就是类的特殊形式,用class 能够定义各类类型,delegate 能够定义各类类型的委托。定义好的委托至关于C语言中定义的函数类型,两种定义和使用方式以下:安全
C:typedef void FunSample(int pValue); 符合的函数实例:void setValue(int pValue);多线程
FunSample *pf=setValue; 函数名为函数在代码段中的起始地址,此处定义指针变量标识该地址架构
C#:delegate void FunSample(int pValue);符合的函数实例:void setValue(int pValue);异步
FunSample fs=new FunSample(setValue);固然还有其余的初始化方式,委托是一种类,因此能够用关键字new进行实例化函数
再简单说下Windows的GUI架构,目前Windows GUI的处理都是在一个单独的GUI线程中的,每一个窗口有一个消息队列,消息循环和处理消息的窗口过程。更详细的内容可自行查看相关介绍。要构建一个响应友好,操做复杂的界面程序,不可避免的会使用到多线程。好比一个文件读取操做,操做在GUI线程中执行,文件很大时,IO读取势必会很耗时,此时整个GUI线程将卡在文件读取部分,其余的消息,如移动,点击等将不会获得及时的处理,界面会卡住。所以,一些耗时操纵须要分离到另外的工做线程中。this
多线程的一个问题是对共享资源的竞争,GUI中会建立不少控件,GUI线程自己去操做这些控件不会有问题,可是其余的线程同时操做这些控件而不去控制,就会出现竞争和不可预料的结果。因此,.net 中对非GUI线程操做GUI控件是作了限制的,同时也提供了几种方式来容许外部线程对GUI控件的安全操做。本篇只介绍其中的一种方式:即经过控件的Invoke 和BeginInvoke 两个函数将非GUI线程对控件的操做交给GUI线程去处理。spa
.net WinForm程序中的控件都实现了ISynchronizeInvoke接口,其中主要就是Invoke和BeginInvoke两个方法。Invoke为同步阻塞,BeginInvoke 为异步非阻塞。.net
Invoke 的解释为在建立当前控件的基础线程上同步执行给定的委托,BeginInvoke相同,只是要异步执行给定的委托。简单来讲,就是将要执行的操做交给当前控件所属的GUI线程去处理。下面简单举一个不断更新GUI界面标签 lbValue 值的实现步骤:线程
private void SetLBValue(int count);指针
根据以上函数能够定义一个简单的委托 delegate void ChangeValueHandler(int count);
private void SetLBValue(int count)
{
this.lbValue.Text = string.Format("当前值:{0}", count);
}
// 开始更新的线程
private void button1_Click(object sender, EventArgs e)
{
new Thread(run).Start();
}
private void run()
{
int count=0;
while (true)
{
this.lbValue.Invoke(new ChangeValueHandler(SetLBValue), count++);
}
}
以上的代码还有值得回味的地方,后续补充完整!