当一个事件被触发时,订阅该事件的方法将在触发该事件的线程中执行。也就是说,订阅该事件的方法在触发事件的线程中同步执行。由此,存在一个问题:如果订阅事件的方法执行时间很长,触发事件的线程被阻塞,长时间等待方法执行完毕。这样,不仅影响后续订阅事件方法的执行,也影响主线程及时响应用户的其他请求。如何处理这个问题呢?讲到此,我想您已经想到了,那就是异步事件调用。
怎样实现异步事件调用呢?如果您对事件比较了解的话,您应该知道事件的本质其实是一种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都不相同,由此可知我们实现了异步事件调用