用委托(Delegate)的BeginInvoke和EndInvoke方法操作线程
在 C#中使用线程的方法很多,使用委托的BeginInvoke和EndInvoke方法就是其中之一。
BeginInvoke方法可以使用线程异步地执行委托所指向的方法。然后通过EndInvoke方法获得方法的返回值(EndInvoke方法的返回值就是被调用方法的返回值),或是确定方法已经被成功调用。我们可以通过四种方法从EndInvoke方法来获得返回值。
使用IAsyncResult asyncResult属性来判断异步调用是否完成
虽然上面的方法可以很好地实现异步调用,但是当调用EndInvoke方法获得调用结果时,整个程序就象死了一样,这样做用户的感觉并不会太好,因此,我们可以使用asyncResult来判断异步调用是否完成,并显示一些提示信息。这样做可以增加用户体验。代码如下:
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading;
namespace MyThread
{
class Program
{
private static int newTask( int ms)
{
Console.WriteLine( " 任务开始 " );
Thread.Sleep(ms);
Random random = new Random();
int n = random.Next( 10000 );
Console.WriteLine( " 任务完成 " );
return n;
}
private delegate int NewTaskDelegate( int ms);
static void Main( string [] args)
{
NewTaskDelegate task = newTask;
IAsyncResult asyncResult = task.BeginInvoke( 2000 , null , null );
while ( ! asyncResult.IsCompleted)
{
Console.Write( " * " );
Thread.Sleep( 100 );
}
// 由于异步调用已经完成,因此, EndInvoke会立刻返回结果
int result = task.EndInvoke(asyncResult);
Console.WriteLine(result);
}
}
}
}
使用WaitOne方法等待异步方法执行完成
使用WaitOne方法是另外一种判断异步调用是否完成的方法。代码如下
while ( ! asyncResult.AsyncWaitHandle.WaitOne( 100 , false))
{
Console.Write( " * " );
}
使用回调方式返回结果
上面介绍的几种方法实际上只相当于一种方法。这些方法虽然可以成功返回结果,也可以给用户一些提示,但在这个过程中,整个程序就象死了一样(如果读者在 GUI程序中使用这些方法就会非常明显),要想在调用的过程中,程序仍然可以正常做其它的工作,就必须使用异步调用的方式。下面我们使用GUI程序来编写一个例子,代码如下:
private delegate string MyMethod(string s);
private bool mebool = false;
private string method(string s)
{
int i = 0;
while (!mebool)
{
i++;
if (i > 5) mebool = true;
Thread.Sleep(1000);
}
s = "bbbbbbbbbbbbbbbb";
return s;
}
private void MethodCompleted(IAsyncResult asyncResult)
{
if (asyncResult == null) return;
param = (asyncResult.AsyncState as MyMethod).EndInvoke(asyncResult).ToString();
MessageBox.Show("线程结束");
Object[] list = { this, System.EventArgs.Empty };
this.textBox1.BeginInvoke(new EventHandler(LabelShow), list);
}
string param = "";
protected void LabelShow(Object o, EventArgs e)
{
this.textBox1.Text = param;
}
private void button1_Click(object sender, EventArgs e)
{
param="aaaaaaaaaaaaa";
MyMethod my = method;
textBox1.Text = param;
IAsyncResult asyncResult = my.BeginInvoke(param,MethodCompleted, my);
}
由于上面的代码通过异步的方式访问的form上的一个textbox,因此,需要按ctrl+f5运行程序(不能直接按F5运行程序,否则无法在其他线程中访问这个textbox,