当一个事件被触发时,订阅该事件的方法将在触发该事件的线程中执行。也就是说,订阅该事件的方法在触发事件的线程中同步执行。由此,存在一个问题:如果订阅事件的方法执行时间很长,触发事件的线程被阻塞,长时间等待方法执行完毕。这样,不仅影响后续订阅事件方法的执行,也影响主线程及时响应用户的其他请求。如何处理这个问题呢?讲到此,我想您已经想到了,那就是异步事件调用。
怎样实现异步事件调用呢?如果您对事件比较了解的话,您应该知道事件的本质其实是一种MulticastDelegate(多播委托)。MulticastDelegate类提供了一个GetInvocationList方法,该方法返回此多播委托的委托调用数组。利用该方法就能实现我们的异步事件调用功能。
示例代码:
using System;
using System.Threading;
using System.Runtime.Remoting.Messaging;
namespace ProcessTest
{
class Program
{
//定义一个事件
public static event EventHandler<EventArgs> OnEvent;
//方法1
static void Method1(object sender, EventArgs e)
{
//显示执行该方法的线程ID
Console.WriteLine("调用Method1的线程ID为:{0}", Thread.CurrentThread.ManagedThreadId);
Thread.Sleep(1000);
}
//方法2
static void Method2(object sender, EventArgs e)
{
Console.WriteLine("调用Method2的线程ID为:{0}", Thread.CurrentThread.ManagedThreadId);
Thread.Sleep(1000);
}
static void Main(string[] args)
{
//显示主线程ID
System.Console.WriteLine("主线程ID为:{0}", Thread.CurrentThread.ManagedThreadId);
//将Method1和Method2注册到事件中
OnEvent += new EventHandler<EventArgs>(Method1);
OnEvent += new EventHandler<EventArgs>(Method2);
//下面的代码实现事件的异步调用
//获取事件中的多路委托列表
Delegate[] delegAry = OnEvent.GetInvocationList();
//遍历委托列表
foreach (EventHandler<EventArgs> deleg in delegAry)
{
//异步调用委托
deleg.BeginInvoke(null, EventArgs.Empty, null, null);
}
System.Console.ReadKey();
}
}
}
代码执行结果如下图:
注意观察运行结果就可发现,主线程ID、执行Method1函数的线程ID、执行Method2函数的线程ID都不相同,由此可知我们实现了异步事件调用。