我有一个函数void Validate()
,它包含了一个窗口的所有验证逻辑。
我不能简单地将其注册为事件处理程序,因为它不接受事件处理程序委托所需的参数。另外,不同类型的控件有不同的签名,所以我不能只让Validate
匹配一个签名而忽略它们的内容。
下面是我设置的一个小例子
txt1.TextChanged += Validate_TextChange;
password1.PasswordChanged += Validate_RoutedEvent;
txt2.TextChanged += Validate_TextChange;
txt3.TextChanged += Validate_TextChange;
password2.PasswordChanged += Validate_RoutedEvent;
txt4.TextChanged += Validate_TextChange;
void Validate_RoutedEvent(object sender, RoutedEventArgs e)
{
ValidateOptions();
}
void Validate_TextChange(object sender, TextChangedEventArgs e)
{
ValidateOptions();
}
public void ValidateOptions()
{
//Actual validation here
}
这只是2个例子,更多的控件可以有更多的签名。有没有更好的方法让所有事件处理程序调用一个函数的情况下,我不关心被传递的参数?
编辑:我喜欢Jon提出的添加参数并简单忽略它们的选项。这确实解决了大部分问题,但是每当我想直接调用这个函数时,比如手动触发验证,我就必须包含虚拟参数来满足编译器。ValidateOptions(this, new EventArgs())
Dan提出的使用匿名函数的建议可以处理这个问题,但是在关联事件处理程序时就不那么干净了。
似乎两种解决方案都不能让你在忽略签名的同时将函数注册为事件处理程序,同时保留调用函数而不创建虚拟参数的能力,但是有多种方法可以非常接近。
编辑:
下面是更新后的示例,实现了Jon的通用事件处理解决方案,但保留了一个可以直接调用的0参数函数
txt1.TextChanged += ValidationEvent;
password1.PasswordChanged += ValidationEvent;
txt2.TextChanged += ValidationEvent;
txt3.TextChanged += ValidationEvent;
password2.PasswordChanged += ValidationEvent;
txt4.TextChanged += ValidationEvent;
//Single event handler accepting EventArgs, which is the base class
//for all more-specific event classes
void ValidationEvent(object sender, EventArgs e)
{
//Ignores arguments and calls 0 argument ValidateOptions
ValidateOptions();
}
//0 argument function that performs actual validation and can be called
//directly from other functions without the need to pass in a fake sender
//and eventargs parameter
public void ValidateOptions()
{
//Actual validation here
}
您可以有一个忽略参数的方法:
public void ValidateOptions(object sender, EventArgs e)
{
}
编译器将允许从ValidateOptions
方法组转换为任何委托类型,遵循正常的事件模式,例如第一个参数是sender
,第二个参数是EventArgs
派生的某种类型。
您可以使用lambda表达式来减少为包装器编写的代码量,但是,在幕后,它仍然在创建包装器方法来丢弃参数。
txt3.TextChanged += (s,e) => ValidateOptions();
password2.PasswordChanged += (s,e) => ValidateOptions();
txt4.TextChanged += (s,e) => ValidateOptions();
这个处理程序为框架控件提供了一个回调,参数'e'被保留以供使用…
private static void EventHandlerSink(object sender, dynamic e)
{
}