C#多线程编程之:异步方法调用
时间:2011-04-04 来源:辛勤的代码工
当一个线程调用方法后,直到方法执行完毕,线程才继续执行,这种方法被称为同步方法。然而,有些方法执行时间可能非常长,比如串口操作或访问网络,这样线程被阻塞,而无法响应用户的其他请求。这种情况通常是无法忍受的,所以这时我们应该使用异步方法。
异步方法的原理是,在方法调用前为异步方法指定一个回调函数,方法调用后被线程池中的一个线程接管,执行该方法。主线程立即返回,继续执行其他工作或响应用户请求。如果异步方法执行完毕,回调函数被自动执行,以处理异步方法的调用结果。
如何实现异步方法呢?C#通过异步委托调用BeginInvoke和EndInvoke方法来实现异步方法。
BeginInvoke方法原型:
IAsyncResult BeginInvoke(......, AsyncCallback callback, object o);
......表示异步委托中定义的参数列表。
AsyncCallback参数是一个用于回调函数的委托,它的原型为:
public delegate void AsyncCallback(IAsyncResult ar)。其中IAsyncResult参数用于包装异步方法的执行结果。
Object参数用于在主线程与回调函数间传递一些附加信息,如同步信息。
EndInvoke方法原型:
xxx EndInvoke(IAsyncResult result);
xxx表示异步委托原型中定义的返回数据类型,IAsyncResult用于包装异步方法的执行结果。
这么看着是不是有点迷糊?看个例子就明白了:
using System;
using System.Threading;
using System.Runtime.Remoting.Messaging;
namespace ProcessTest
{
class Program
{
//异步调用执行完成同步信号
static AutoResetEvent ev = new AutoResetEvent(false);
//定义委托
public delegate int Deleg(int a, int b);
static int WriteSum(int a, int b)
{
//显示当前线程ID号及Sum值
Console.WriteLine("执行WriteSum的线程ID为:{0},Sum = {1}", Thread.CurrentThread.ManagedThreadId, a + b);
return a + b;
}
//回调函数
static void SumDone(IAsyncResult async)
{
//等待1秒,模拟线程正在执行其他工作
Thread.Sleep(1000);
//async中包装了异步方法执行的结果
//从操作结果async中还原委托
Deleg proc = ((AsyncResult)async).AsyncDelegate as Deleg;
//获取异步方法的执行结果
int sum = proc.EndInvoke(async);
//显示结果
Console.WriteLine("执行SumDone的线程ID为:{0},Sum = {1}", Thread.CurrentThread.ManagedThreadId, sum);
//使用AsnycState属性获取主线程中传入的同步信号
//释放同步信号表示异步调用已完成
((AutoResetEvent)async.AsyncState).Set();
}
static void Main(string[] args)
{
//创建一个委托
Deleg proc = new Deleg(WriteSum);
//采用异步方式调用委托
//指定SumDone为异步操作完成后的回调函数
//指定ev为object参数,用于同步回调函数与主线程间操作
IAsyncResult async = proc.BeginInvoke(10, 10, SumDone, ev);
Console.WriteLine("主线程ID号为:{0},异常操作已开始执行,正等待操作完成。", Thread.CurrentThread.ManagedThreadId);
//等待异步操作完成
ev.WaitOne();
Console.WriteLine("异常操作已完成!");
System.Console.ReadKey();
}
}
}
该示例首先定义了一个委托,然后使用异步方法执行该委托,并且为异步方法指定SumDone为回调函数,指定ev为附加信息用于回调函数与主线程间的同步操作。异步操作的执行结果在回调函数中可通过IAsyncResult参数取出。
下图是程序的运行结果:
注意观察运行结果,异步方法和回调函数是在同一步线程执行。