当一个事件被触发时,订阅该事件的方法将在触发该事件的线程中执行。也就是说,订阅该事件的方法在触发事件的线程中同步执行。由此,存在一个问题:如果订阅事件的方法执行时间很长,触发事件的线程被阻塞,长时间等待方法执行完毕。这样,不仅影响后续订阅事件方法的执行,也影响主线程及时响应用户的其他请求。如何处理这个问题呢?讲到此,我想您已经想到了,那就是异步事件调用。

  怎样实现异步事件调用呢?如果您对事件比较了解的话,您应该知道事件的本质其实是一种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都不相同,由此可知我们实现了异步事件调用。

posted on 2011-04-04 16:46  辛勤的代码工  阅读(8510)  评论(0编辑  收藏  举报