C ++ Boost Enable_If вопрос
Вопрос
У меня есть способ упростить следующие утверждения? (вероятно, используя boost::enable_if
).
У меня простая структура классов - Base
Базовый класс, Derived1
, Derived2
наследовать от Base
.
У меня есть следующий код:
template <typename Y> struct translator_between<Base, Y> {
typedef some_translator<Base, Y> type;
};
template <typename Y> struct translator_between<Derived1, Y> {
typedef some_translator<Derived1, Y> type;
};
template <typename Y> struct translator_between<Derived2, Y> {
typedef some_translator<Derived2, Y> type;
};
Я хочу написать то же самое заявление, используя один шаблон специализации translator_between
.
Пример (псевдокод) того, что я хочу иметь возможность писать:
template <typename Class, typename Y>
ONLY_INSTANTIATE_THIS_TEMPLATE_IF (Class is 'Base' or any derived from 'Base')
struct translator_between<Class, Y> {
typedef some_translator<Class, Y> type;
};
Любой способ добиться этого использования boost::enable_if
а также boost::is_base_of
?
Решение
Во-первых, вам придется выбрать ваш выбор среди:
is_base_of
is_convertible
оба можно найти в <boost/type_traits.hpp>
, Последнее быть более допустимым.
Если вы просто предотвратите реализацию этого типа для некоторой комбинации, то используйте статическое утверждение:
// C++03
#include <boost/mpl/assert.hpp>
template <typename From, typename To>
struct translator_between
{
BOOST_MPL_ASSERT((boost::is_base_of<To,From>));
typedef translator_selector<From,To> type;
};
// C++0x
template <typename From, typename To>
struct translator_between
{
static_assert(boost::is_base_of<To,From>::value,
"From does not derive from To");
typedef translator_selector<From,To> type;
};
Так как здесь нет разрешения перегрузки, вам не нужно enable_if
.
Другие советы
Я не думаю boost::enable_if
Помогает, потому что Sfinae, кажется, скорее, о выборе между функциональными перегрузками.
Вы можете, конечно, использовать шаблоны с bool
Параметры для уточнения выбора:
#include <boost/type_traits.hpp>
class Base {};
class Derived : public Base {};
template <class A, class B>
struct some_translator {};
template <typename A, typename B, bool value>
struct translator_selector; //perhaps define type as needed
template <typename A, typename B>
struct translator_selector<A, B, true>
{
typedef some_translator<A, B> type;
};
template <typename A, typename B>
struct translator_between
{
typedef typename translator_selector<A, B, boost::is_base_of<Base, A>::value>::type type;
};
int main()
{
translator_between<Base, int>::type a;
translator_between<Derived, int>::type b;
translator_between<float, int>::type c; //fails
}
Вы можете использовать Anable_If и этот макрос здесь, чтобы сделать его более читаемым:
#define CLASS_REQUIRES(...) typename boost::enable_if<boost::mpl::and_<__VA_ARGS__, boost::mpl::bool_<true> > >::type
Тогда вы можете определить ваш класс, как это:
template <typename Class, typename Y, class Enable =
CLASS_REQUIRES(boost::is_base_of<Class, Y>)>
struct translator_between {
typedef some_translator<Class, Y> type;
};