今天遇到一个问题,就是希望能够动态调用事件。传统的思路是,通过Reflection.EventInfo获得事件的信息,然后使用GetRaiseMethod方法获得事件被触发后调用的方法,再使用MethodInfo.Invoke来调用以实现事件的动态调用。
很不幸,Reflection.EventInfo.GetRaiseMethod方法始终返回null。这是因为,C#编译器在编译并处理由event关键字定义的事件时,根本不会去产生有关RaiseMethod的元数据信息,因此GetRaiseMethod根本无法获得事件触发后的处理方法。Thottam R. Sriram 在其Using SetRaiseMethod and GetRaiseMethod and invoking the method dynamically 一文中简要介绍了这个问题,并通过Reflection.Emit相关的方法来手动生成RaiseMethod,最后使用常规的GetRaiseMethod来实现事件触发后的方法调用。这种做法比较繁杂。以下代码是一个简单的替代方案,同样可以实现事件的动态调用:
- public event EventHandler<EventArgs> MyEventToBeFired;
- public void FireEvent(Guid instanceId, string handler)
- {
- // Note: this is being fired from a method with in the same class that defined the event (i.e. "this").
- EventArgs e = new EventArgs(instanceId);
- MulticastDelegate eventDelagate = (MulticastDelegate)this
- .GetType()
- .GetField(handler, BindingFlags.Instance | BindingFlags.NonPublic)
- .GetValue(this);
- Delegate[] delegates = eventDelagate.GetInvocationList();
- foreach (Delegate dlg in delegates)
- {
- dlg.Method.Invoke( dlg.Target, new object[] { this, e } );
- }
- }
- FireEvent(new Guid(), "MyEventToBeFired");