不能将继承的类作为SFINAE的基础进行特征化

时间:2013-03-25 13:55:01

标签: c++ inheritance c++11 traits sfinae

我创建了一个container_traits类来检查容器是否为std::array。 但它无法捕获从std::array继承的容器。 任何解决方案?

#include <vector>
#include <array>
#include <iostream>

using namespace std;

template<typename C>
struct container_traits { constexpr static bool is_array = false; };
template<typename T, size_t S>
struct container_traits<std::array<T,S>> { constexpr static bool is_array = true; };

template<typename T, size_t S>
struct A : public std::array<T,S> {};

int main()
{
    cout << container_traits<A<int, 5>>::is_array << endl;              // must return 1
    cout << container_traits<std::array<int, 10>>::is_array << endl;    // must return 1
    cout << container_traits<std::vector<int>>::is_array << endl;       // must return 0
    return 0;
}

2 个答案:

答案 0 :(得分:7)

您可以使用std::is_base_of

template <typename C>
struct container_traits {
  constexpr static bool is_array = false;
};

template <template <typename, std::size_t> class C,
          typename T, std::size_t N>
struct container_traits<C<T,N>> {
  constexpr static bool is_array = std::is_base_of<std::array<T,N>, C<T,N>>::value;
};

See it in action

它基本上适用于任何带有两个参数的模板类型。在该专业化中,is_array的值由std::is_base_of确定。

答案 1 :(得分:6)

作者注:请不要投票支持此解决方案。投票支持sftrabbit,因为它比我的更好。我不会删除,因为虽然有缺陷,但有些人认为这个想法很有意思。

这会得到您想要的结果:

template<typename C>
struct container_traits {

private:

   template <typename T, size_t N>
   static char (&is_array_helper(const std::array<T, N>&))[1];

   static char (&is_array_helper(...))[2];

public:

    constexpr static bool is_array =
        sizeof(is_array_helper(std::declval<C>())) == sizeof(char[1]);

};

注意

值得一提的是,对于派生自std::array<T, N>(例如A)的类,继承必须为public,否则代码无法编译。要解决此问题,可以使用std::is_base_of或应用一些SFINAE技术。但是,关于可访问性的SFINAE规则已经从C ++ 03改为C ++ 11,而且当我刚刚测试时,一些主要编译器没有实现新规则。也许他们现在就做。

我建议问题的作者等一下,看看是否会出现更好的解决方案(我希望这会发生)。