动态事件订阅和1个处理程序

时间:2013-03-15 16:03:16

标签: c# events reflection

我已经看到了几个答案,但不知怎的,我不能让我的工作。我想动态使用各种控件的任何事件(文本框,复选框,按钮等),最好将它们分配给一个事件处理程序。应在运行时分配处理程序。此外,我想在处理程序中知道哪个事件触发了处理程序。

我得到了部分工作。使用lambda表达式我调用我的处理程序(EventAssistant)并传递一个包含事件名称的额外参数(命令)。它适用于使用EventHandler类型的事件。但是,它不适用于期望不同处理程序的事件,例如类型MouseEventHandler。它无法在AddEventHandler订阅。

private void RegisterEventHandlers(Control ctl)
{
  foreach (Command command in CommandList)
  {
    EventInfo eventInfo = ctl.GetType().GetEvent(command.Name);
    EventHandler handler = (sender, args) =>
    {
      EventAssistant(sender, args, command);
    };
    eventInfo.AddEventHandler(ctl, handler);
  }
}

public void EventAssistant(object sender, EventArgs e, Command c)
{
  //do lots of other fun stuff
}

基于C# passing extra parameters to an event handler?


作为替代方案,我尝试使用表达式树来解决问题,如下所示:Why am I getting an Argument exception when creating event handler dynamically? 显然,可以从EventInfo中检索EventHandlerType并在lambda表达式中使用。

但是,无论我做什么,我总是会得到一个InvalidOperationException“Lambda参数不在范围内”。

private void RegisterEventHandlers(Control ctl)
{
  foreach (Command command in CommandList)
  {
    EventInfo eventInfo = ctl.GetType().GetEvent(command.Name);

    var sender = Expression.Parameter(typeof(object), "sender");
    var e = Expression.Parameter(typeof(EventArgs), "e");
    var c = Expression.Parameter(typeof(Command), "command");
    Expression[] arg = new Expression[] { sender, e, c };
    MethodInfo mi = this.GetType().GetMethod("EventAssistant");
    var body = Expression.Call(Expression.Constant(this), mi, arg);
    var lambda = Expression.Lambda(eventInfo.EventHandlerType, body, sender, e);

    eventInfo.AddEventHandler(ctl, lambda.Compile());
  }
}

我在表达树上做错了什么?

此外,第一段代码看起来更干净。是否有可能使用第一个代码示例得到我想要的东西?

1 个答案:

答案 0 :(得分:1)

在您的第二次尝试中,变量c不应该是ParameterExpression,而是ConstantExpression,而值设置为当前command。使用当前代码,您将创建一个处理程序,基本上如下所示:

(_sender, _e) => this.EventAssistant(_sender, _e, _c)
// The expression expects "_c" to be a parameter of the lambda, which is why
// you're getting that exception

但是,如果你改变了

var c = Expression.Parameter(typeof(Command), "command");

var c = Expression.Constant(command);

生成的代码将按预期显示(并且当然可以工作):

(_sender, _e) => this.EventAssistant(_sender, _e, command)