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