Нет вывода типа с универсальным методом расширения - PullRequest
12 голосов
/ 24 августа 2011

У меня есть следующий метод:

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.
}

Теперь у меня есть следующие проблемы:

  1. Этот метод расширения отображается на всех типах, даже string.
  2. Я не могу написать 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;
                }
            }
        }
    }
}

Чтобы убедиться, что этот метод можно вызывать только с полностью настроенными параметрами, он принимает только ConfiguredEventInvocatorParameters<T>, который получается из EventInvocatorParameters<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();

Ответы [ 4 ]

19 голосов
/ 24 августа 2011

Вывод типа общего метода намеренно не делает какие-либо выводы из ограничений. Скорее вычеты производятся из аргументов и формальных параметров , а затем аргументы выведенного типа проверяются на соответствие ограничениям.

Для подробного обсуждения некоторых вопросов проектирования, связанных с ограничениями и сигнатурами методов, в том числе несколько десятков человек, говорящих мне, что я ошибаюсь, полагая, что существующий дизайн является разумным, см. Мою статью на эту тему:

http://blogs.msdn.com/b/ericlippert/archive/2009/12/10/constraints-are-not-part-of-the-signature.aspx

3 голосов
/ 24 августа 2011

Для тех, кто заинтересован, на данный момент я решил исходную проблему (свободный вызов API) с помощью общей иерархии классов. Это в основном ответ Hightechrider на стероиды.

public abstract class EventInvocatorParametersBase
    <TEventInvocatorParameters, TEventArgs>
    where TEventArgs : EventArgs
    where TEventInvocatorParameters :
        EventInvocatorParametersBase<TEventInvocatorParameters, TEventArgs>

{
    protected EventInvocatorParametersBase(
        EventHandler<TEventArgs> eventHandler,
        Func<Exception, bool> exceptionHandler,
        Func<TEventArgs, bool> breakCondition)
    {
        EventHandler = eventHandler;
        ExceptionHandler = exceptionHandler;
        BreakCondition = breakCondition;
    }

    protected EventInvocatorParametersBase(
        EventHandler<TEventArgs> eventHandler)
        : this(eventHandler, e => false, e => false)
    {
    }

    public Func<TEventArgs, bool> BreakCondition { get; set; }
    public EventHandler<TEventArgs> EventHandler { get; set; }
    public Func<Exception, bool> ExceptionHandler { get; set; }

    public TEventInvocatorParameters Until(
        Func<TEventArgs, bool> breakCondition)
    {
        BreakCondition = breakCondition;
        return (TEventInvocatorParameters)this;
    }

    public TEventInvocatorParameters WithExceptionHandler(
        Func<Exception, bool> exceptionHandler)
    {
        ExceptionHandler = exceptionHandler;
        return (TEventInvocatorParameters)this;
    }

    public ConfiguredEventInvocatorParameters<TEventArgs> With(
        object sender, 
        TEventArgs eventArgs)
    {
        return new ConfiguredEventInvocatorParameters<TEventArgs>(
            EventHandler, ExceptionHandler, BreakCondition,
            sender, eventArgs);
    }
}

public class EventInvocatorParameters<T> :
    EventInvocatorParametersBase<EventInvocatorParameters<T>, T>
    where T : EventArgs
{
    public EventInvocatorParameters(EventHandler<T> eventHandler)
        : base(eventHandler)
    {
    }
}

public class ConfiguredEventInvocatorParameters<T> :
    EventInvocatorParametersBase<ConfiguredEventInvocatorParameters<T>, T>
    where T : EventArgs
{
    public ConfiguredEventInvocatorParameters(
        EventHandler<T> eventHandler,
        Func<Exception, bool> exceptionHandler,
        Func<T, bool> breakCondition, object sender,
        T eventArgs)
        : base(eventHandler, exceptionHandler, breakCondition)
    {
        EventArgs = eventArgs;
        Sender = sender;
    }

    public ConfiguredEventInvocatorParameters(EventHandler<T> eventHandler,
                                              object sender,
                                              T eventArgs)
        : this(eventHandler, e => false, e => false, sender, eventArgs)
    {
    }

    public T EventArgs { get; private set; }
    public object Sender { get; private set; }
}

public static class EventExtensions
{
    public static EventInvocatorParameters<TEventArgs> Until<TEventArgs>(
        this EventHandler<TEventArgs> eventHandler,
        Func<TEventArgs, bool> breakCondition)
        where TEventArgs : EventArgs
    {
        return new EventInvocatorParameters<TEventArgs>(eventHandler).
            Until(breakCondition);
    }

    public static EventInvocatorParameters<TEventArgs> 
        WithExceptionHandler<TEventArgs>(
            this EventHandler<TEventArgs> eventHandler,
            Func<Exception, bool> exceptionHandler)
        where TEventArgs : EventArgs
    {
        return
            new EventInvocatorParameters<TEventArgs>(eventHandler).
                WithExceptionHandler(exceptionHandler);
    }

    public static ConfiguredEventInvocatorParameters<TEventArgs>
        With<TEventArgs>(
            this EventHandler<TEventArgs> eventHandler, object sender,
            TEventArgs eventArgs)
        where TEventArgs : EventArgs
    {
        return new ConfiguredEventInvocatorParameters<TEventArgs>(
            eventHandler, sender, eventArgs);
    }
}

Это позволяет вам написать код, подобный этому:

Fire.Event(EventName.WithExceptionHandler(e => false)
                    .Until(e => false).With(this, EventArgs.Empty));
Fire.Event(EventName.With(this, EventArgs.Empty));
Fire.Event(EventName.WithExceptionHandler(e => false)
                    .With(this, EventArgs.Empty).Until(e => false));
Fire.Event(EventName.With(this, EventArgs.Empty)
                    .WithExceptionHandler(e => false).Until(e => false));

Но это не позволяет вам написать это, потому что не вся необходимая информация (eventArgs и sender) была предоставлена:

Fire.Event(EventName.Until(e => false));
Fire.Event(EventName);
1 голос
/ 24 августа 2011

Есть ли какая-то причина, по которой вам нужно использовать метод расширения?Если вы ставите Until на EventInvocatorParameters<T> класс, вы можете избежать обеих упомянутых проблем:

public class EventInvocatorParameters<T>
    where T : EventArgs
{
    public Func<T, bool> BreakCondition { get; set; }
    // Other properties used below omitted for brevity.

    public EventInvocatorParameters<T> Until (Func<T, bool> breakCond)
    {
        this.BreakCondition = breakCond;
        return this;
    }
}
0 голосов
/ 24 августа 2011

Немного отговорки, я знаю, но вы рассматривали возможность использования Rx вместо того, чтобы изобретать то, что вы пытаетесь сделать?

...