Я хотел бы написать метод расширения для универсального типа с дополнительными параметрами универсального типа.У меня уже есть рабочий код, но результат мне не нравится, потому что пользователь должен повторно ввести параметры универсального типа существующего типа.
У меня есть конкретный пример, но имейте в виду, что этоэто общая проблема.Я ценю отзывы о конкретной проблеме, но я ищу общие решения.
Для класса типа Action<T>
добавьте метод расширения с сигнатурой, подобной этой:
Func<T, TResult> ToFunc<TResult>();
Вот мой рабочий код:
public static class DelegateExtensions
{
public static Func<T, TResult> ToFunc<T, TResult>(this Action<T> action)
{
return arg => { action(arg); return default(TResult); };
}
}
Но использование воняет:
public void TakesAFunc(Func<int, float> someFunc) { /* ... */ }
// ...
Action<int> someIntAction = /* ... */;
TakesAFunc(someIntAction.ToFunc<int, float>());
В этом примере универсальный параметр int
является единственным допустимым значением, поэтомуэто вызывает ненужное дублирование кода.
Вывод типа здесь не работает.Я думаю, это потому, что вы не можете вывести общий параметр через тип возвращаемого значения.
Этот код решит проблему, но, к сожалению, не работает:
public static class DelegateExtensions<T>
{
public static Func<T, TResult> ToFunc<TResult>(this Action<T> action)
{
return arg => { action(arg); return default(TResult); };
}
}
Использование будетТочно так же, как и следовало ожидать:
public void TakesAFunc(Func<int, float> someFunc) { /* ... */ }
// ...
Action<int> someIntAction = /* ... */;
TakesAFunc(someIntAction.ToFunc<float>());
Я заметил, что System.Linq.Queryable
работает так же, как мой первый кусок кода, хотя обычно для него не нужны дополнительные параметры типа, поэтому вывод типа работает.
Есть ли какой-нибудь известный способ обойти эти дубликаты параметров универсального типа?Одна мысль, которая приходит на ум, - это генераторы кода или какие-то макросы, но я не могу придумать, как бы это сделать чисто.