c – 检查模板类的某些专门化是否是特定类的基类

现代STL中有std :: is_base_of.它允许我们确定第二个参数是从第一个参数派生的,或者它们是否是相同的类,或者否则确定它们之间是否没有这样的关系.

是否可以确定一个类是从一些具体的模板类派生的,而不区分其专业化涉及的具体实际参数?

说,我们有

template< typename ...types >
struct B {};

template< typename ...types >
struct D : B< types... > {};

是否可以定义类型特征:

template< typename T > is_derived_from_B;

当T是D的任何特殊化并且从std :: false_type派生时,如果T不是从B的任何专业化导出的,那么它来自std :: true_type?

解决方法

如果您可以假设派生类型使用B< Args ...>的公共继承(所以可以上传),那么可以使用以下SFINAE:
namespace detail
{
    template <typename Derived>
    struct is_derived_from_B
    {
        using U = typename std::remove_cv<
                                  typename std::remove_reference<Derived>::type
                                >::type;

        template <typename... Args>
        static auto test(B<Args...>*)
            -> typename std::integral_constant<bool,!std::is_same<U,B<Args...>>::value>;

        static std::false_type test(void*);

        using type = decltype(test(std::declval<U*>()));
    };
}

template <typename Derived>
using is_derived_from_B = typename detail::is_derived_from_B<Derived>::type;

测试:

static_assert(is_derived_from_B<const D<int,char,float>>::value,"!");
static_assert(!is_derived_from_B<int>::value,"!");
static_assert(!is_derived_from_B<B<int,int>>::value,"!");
static_assert(!is_derived_from_B<std::vector<int>>::value,"!");

DEMO 1

可以将其概括为接受任何基类模板:

namespace detail
{
    template <template <typename...> class Base,typename Derived>
    struct is_derived_from_template
    {
        using U = typename std::remove_cv<
                                  typename std::remove_reference<Derived>::type
                                >::type;

        template <typename... Args>
        static auto test(Base<Args...>*)
            -> typename std::integral_constant<bool,Base<Args...>>::value>;

        static std::false_type test(void*);

        using type = decltype(test(std::declval<U*>()));
    };
}

template <template <typename...> class Base,typename Derived>
using is_derived_from_template
                = typename detail::is_derived_from_template<Base,Derived>::type;

测试:

static_assert(is_derived_from_template<B,const D<int,"!");
static_assert(!is_derived_from_template<B,int>::value,B<int,std::vector<int>>::value,"!");

DEMO 2

相关文章

本程序的编译和运行环境如下(如果有运行方面的问题欢迎在评...
水了一学期的院选修,万万没想到期末考试还有比较硬核的编程...
补充一下,先前文章末尾给出的下载链接的完整代码含有部分C&...
思路如标题所说采用模N取余法,难点是这个除法过程如何实现。...
本篇博客有更新!!!更新后效果图如下: 文章末尾的完整代码...
刚开始学习模块化程序设计时,估计大家都被形参和实参搞迷糊...