3

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

我得到了这个部分工作。使用 lambda 表达式,我调用我的处理程序 (EventAssistant) 并传递一个包含事件名称的额外参数 (command)。它适用于使用 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# 将额外参数传递给事件处理程序?


作为替代方案,我尝试使用表达式树来解决问题,如下所示:为什么动态创建事件处理程序时出现参数异常? 显然,可以从 EventInfo 中检索 EventHandlerType 并在 lambda 表达式中使用。

但是,无论我做什么,我总是得到一个 InvalidOperationException “Lambda Parameter not in scope”。

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());
  }
}

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

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

4

1 回答 1

1

在您的第二次尝试中,变量c不应该是 a ParameterExpression,而是 aConstantExpression的值设置为 current 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)
于 2013-03-15T16:44:58.993 回答