зависит от свойств типов передаваемых параметров (наличие/отсутсвие некоторых методов, арифметрический ли это тип, и т.д.)
сейчас пачка специализаций через enable_if
template<typename T>
typename enable_if_t<has_method_foo<T>::value, bool> doSomething(T t)
template<typename T>
typename enable_if_t<has_method_bar<T>::value, bool> doSomething(T t)
и т.д.
Вопрос как добавить поведение по-умолчанию?,
т.е. если другие enable_if не специализировали функцию, то использовать эту реализацию
Я знаю что это можно делать через тот же enable_if, но с исключающими условиями:
template<typename T>
typename enable_if_t<!has_method_bar<T>::value && !has_method_foo<T>::value, bool> doSomething(T t) { doDefault(t); }
Такой способ подходит, до тех пор пока специализаций мало.
Существует ли более удобный способ использовать SFINAE, где можно было бы удобно указать реализацию по-умолчанию?
В подобных случаях можно воспользоваться трюком из Boost.Variant: template<typename T> typename enable_if_t<has_method_foo<T>::value, bool> doSomething_impl(T t,long); template<typename T> typename enable_if_t<has_method_bar<T>::value, bool> doSomething_impl(T t, long); template<typename T> bool doSomething_impl(T t, int) { return doDefault(t); } template<typename T> bool doSomething(T t, int) { return doSomething_impl(t, 1L); } 1L будет пытаться сматчится с функциями принимающими long в первую очередь. Когда все такие функции закончатся - возьмёт doSomething_impl(T t, int). Неиспользуемый параметр у шаблонной функции все современные компиляторы должны выкидывать, так что на производительность никак не повлияет.
Обсуждают сегодня