AddEventHandler使用反射到静态类

时间:2015-11-12 07:29:34

标签: c# reflection event-handling

我需要将新方法附加到静态类的现有事件中。 这是我的猫#39;静态类,不能改变它,因为它是第三方。我需要添加新方法以在同一事件myEvent中触发。 This问题显示了如何为非静态类做到这一点。但是我不确定如何为静态类做。有人帮忙吗?

public static class Cat
{
 internal static event EventHandler myEvent;

public static init()
{
....
....
}

}

class Dog
{
 public static void init()
 {
   EventInfo eventInfo = typeof(Cat).GetEvent("myEvent");

      if (eventInfo != null)
      {
       Delegate handler =  Delegate.CreateDelegate(eventInfo.EventHandlerType, , newMethod); // I'm stuck here
       eventInfo.AddEventHandler(, handler);  // I'm stuck here

     }
 }

  static void newMethod()
  {

  }
}

如果myEvent是公开的,我可以Cat.myEvent+= new EventHandler(newMethod);。但不幸的是,internal

3 个答案:

答案 0 :(得分:1)

你应该添加一些东西。首先,将BindingFlags添加到GetEvent,因为该事件是静态的而非公开的。默认情况下,GetEvent仅返回公共实例事件:

EventInfo eventInfo = typeof(Cat).GetEvent("myEvent", BindingFlags.Static | BindingFlags.NonPublic);

其次,使用CreateDelegate的另一个重载来创建静态方法的委托,该方法具有两个参数。或者使用你的,将null作为第二个参数。

第三,使用null作为AddEventHandler的第一个参数。通常,您将实例作为第一个参数传递,但由于此事件是静态的,因此它不需要实例:

eventInfo.AddEventHandler(null, handler);  

答案 1 :(得分:1)

你没有走得太远,但是当你尝试使用AddEventHandler()时,你会非常努力。即使您尝试使用反射破解它,CLR也会强制执行事件的可访问性。因为它不是public,所以反思并没有帮助。

你必须找到更大的武器。假设事件不是自定义且月亮位于右象限,则C#编译器会生成 add 访问器方法。这是一个名为add_myEvent()的方法。使用反编译器来验证名称,当然它实际上并没有这样命名。

您要做的第一件事是为您的事件处理程序提供正确的签名,该签名与委托类型匹配:

    static void newMethod(object sender, EventArgs e) {
        // etc..
    }

然后你可以像这样编写反射代码:

    public static void init() {
        EventInfo eventInfo = typeof(Cat).GetEvent("myEvent",
            BindingFlags.NonPublic | BindingFlags.Static);
        MethodInfo adder = typeof(Cat).GetMethod("add_myEvent",
            BindingFlags.NonPublic | BindingFlags.Static);
        MethodInfo target = typeof(Dog).GetMethod("newMethod", 
            BindingFlags.NonPublic | BindingFlags.Static);
        Delegate handler =  Delegate.CreateDelegate(eventInfo.EventHandlerType, target);
        adder.Invoke(null, new object[] { handler });
    }

答案 2 :(得分:0)

由于Dog在您的控制之下,因此无需Delegate.CreateDelegate。

此代码按预期工作:

namespace ConsoleApplication1
{

    public static class Cat
    {
        internal static event EventHandler myEvent;

        public static void RaiseEvent()
        {
            if (myEvent != null) myEvent(typeof(Cat), EventArgs.Empty);
        }
    }


    class Dog
    {
        public static void init()
        {
            MethodInfo addInfo = typeof(Cat).GetMethod("add_myEvent", BindingFlags.NonPublic | BindingFlags.Static);
            EventHandler handler = new EventHandler(Handler);
            addInfo.Invoke(null, new object[] { handler });
        }

        static void Handler(object sender, EventArgs e)
        {
            Console.WriteLine("Event fired");
        }
    }
    class Program
    {
        static void Main(string[] args)
        {
            Dog.init();
            Cat.RaiseEvent();
        }
    }
}