我有如下方法:
现在,我有以下问题:
更广泛的背景(回答问题不必要!): 我正在尝试创建一个流畅的接口来调用事件。 基础是这个静态类:
为了确保此方法只能使用完全配置的参数进行调用,它只接受从
以下调用将是有效的:
以下内容是无效的:
为了使其工作,存在名为
如果您对API有任何想法,请让我知道。然而,我想避免以下三件事情:
public static TEventInvocatorParameters Until
<TEventInvocatorParameters, TEventArgs>(this TEventInvocatorParameters p,
Func<TEventArgs, bool> breakCond)
where TEventInvocatorParameters : EventInvocatorParameters<TEventArgs>
where TEventArgs : EventArgs
{
p.BreakCondition = breakCond;
return p;
}
而这个类
public class EventInvocatorParameters<T>
where T : EventArgs
{
public Func<T, bool> BreakCondition { get; set; }
// Other properties used below omitted for brevity.
}
现在,我有以下问题:
- 这个扩展方法会显示在所有类型上,甚至是
string
。 - 我无法编写
new EventInvocatorParameters<EventArgs>(EventABC).Until(e => false);
它告诉我“从用法中无法推断出方法的类型参数。”
更广泛的背景(回答问题不必要!): 我正在尝试创建一个流畅的接口来调用事件。 基础是这个静态类:
public static class Fire
{
public static void Event<TEventArgs>(
ConfiguredEventInvocatorParameters<TEventArgs> parameters)
where TEventArgs : EventArgs
{
if (parameters.EventHandler == null)
{
return;
}
var sender = parameters.Sender;
var eventArgs = parameters.EventArgs;
var breakCondition = parameters.BreakCondition;
foreach (EventHandler<TEventArgs> @delegate in
parameters.EventHandler.GetInvocationList())
{
try
{
@delegate(sender, eventArgs);
if (breakCondition(eventArgs))
{
break;
}
}
catch (Exception e)
{
var exceptionHandler = parameters.ExceptionHandler;
if (!exceptionHandler(e))
{
throw;
}
}
}
}
}
为了确保此方法只能使用完全配置的参数进行调用,它只接受从
EventInvocatorParameters<T>
派生的 ConfiguredEventInvocatorParameters<T>
:public class ConfiguredEventInvocatorParameters<T>
: EventInvocatorParameters<T>
where T : EventArgs
{
public ConfiguredEventInvocatorParameters(
EventInvocatorParameters<T> parameters, object sender, T eventArgs)
: base(parameters)
{
EventArgs = eventArgs;
Sender = sender;
}
public T EventArgs { get; private set; }
public object Sender { get; private set; }
}
以下调用将是有效的:
Fire.Event(EventName.With(sender, eventArgs));
Fire.Event(EventName.With(sender, eventArgs).Until(e => e.Cancel));
Fire.Event(EventName.Until(e => e.Cancel).With(sender, eventArgs));
以下内容是无效的:
// no sender or eventArgs have been specified, i.e. missing call to With(...)
Fire.Event(EventName.Until(e => e.Cancel));
为了使其工作,存在名为
With
的扩展方法,它们接受EventHandler<TEventArgs
或TEventInvocatorParameters
,并返回ConfiguredEventInvocatorParameters<TEventArgs>
。现在,在With
之后的所有调用也需要返回类型ConfiguredEventInvocatorParameters<TEventArgs>
,否则有效调用的第二个示例(以Until
结尾)将不起作用。如果您对API有任何想法,请让我知道。然而,我想避免以下三件事情:
- 仅在运行时失败,如果参数没有完全配置
- 创建倒置语法,例如
EventName.With(...).Until(...).Fire()
- 使用臭名昭著的
Do
方法来启动事物:Fire(EventName).With(...).Until(...).Do();