我创建了一个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;
}
答案 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;
};
它基本上适用于任何带有两个参数的模板类型。在该专业化中,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,而且当我刚刚测试时,一些主要编译器没有实现新规则。也许他们现在就做。
我建议问题的作者等一下,看看是否会出现更好的解决方案(我希望这会发生)。