SFINAE:运算符 [] 上的 decltype
SFINAE: decltype on operator[]
根据答案 and 我正在尝试使用以下内容
template <typename T>
using operator_square_brackets = decltype(&T::operator[]);
它在 visual studio 和
上失败
error C2760: syntax error: expected ')' not ']'
关于如何解决这个问题有什么想法吗?
如果要检测类型是否具有特定函数或重载运算符,则必须调用该函数或运算符。这很重要,因为您可能有多个函数或运算符重载,并且重载解决方案始终取决于调用者。
这里有一个小例子,基于CppCon 2014: Walter E. Brown "Modern Template Metaprogramming: A Compendium, Part II"关于如何检测类型中的operator[]
。
我不知道为什么 VC 会给你一个看起来更像是解析错误的奇怪错误。我本以为会出现类似 »无法解决对重载函数的引用;你是想打电话给它吗?«。
#include <string>
#include <type_traits>
#include <vector>
// in C++17 std::void_t
template < typename... >
using void_t = void;
template < typename T, typename Index >
using subscript_t = decltype(std::declval<T>()[std::declval<Index>()]);
template < typename, typename Index = size_t, typename = void_t<> >
struct has_subscript : std::false_type {};
template < typename T, typename Index >
struct has_subscript< T, Index, void_t< subscript_t<T,Index> > > : std::true_type {};
struct A
{
void operator[](size_t) {}
};
struct B {};
int main ()
{
static_assert(has_subscript< std::vector<int> >::value == true , "!");
static_assert(has_subscript< std::vector<double> >::value == true , "!");
static_assert(has_subscript< A >::value == true , "!");
static_assert(has_subscript< A, std::string >::value == false, "!");
static_assert(has_subscript< B >::value == false, "!");
static_assert(has_subscript< double[5] >::value == true , "!");
static_assert(has_subscript< double* >::value == true , "!");
static_assert(has_subscript< double >::value == false, "!");
}
根据答案
template <typename T>
using operator_square_brackets = decltype(&T::operator[]);
它在 visual studio 和
上失败error C2760: syntax error: expected ')' not ']'
关于如何解决这个问题有什么想法吗?
如果要检测类型是否具有特定函数或重载运算符,则必须调用该函数或运算符。这很重要,因为您可能有多个函数或运算符重载,并且重载解决方案始终取决于调用者。
这里有一个小例子,基于CppCon 2014: Walter E. Brown "Modern Template Metaprogramming: A Compendium, Part II"关于如何检测类型中的operator[]
。
我不知道为什么 VC 会给你一个看起来更像是解析错误的奇怪错误。我本以为会出现类似 »无法解决对重载函数的引用;你是想打电话给它吗?«。
#include <string>
#include <type_traits>
#include <vector>
// in C++17 std::void_t
template < typename... >
using void_t = void;
template < typename T, typename Index >
using subscript_t = decltype(std::declval<T>()[std::declval<Index>()]);
template < typename, typename Index = size_t, typename = void_t<> >
struct has_subscript : std::false_type {};
template < typename T, typename Index >
struct has_subscript< T, Index, void_t< subscript_t<T,Index> > > : std::true_type {};
struct A
{
void operator[](size_t) {}
};
struct B {};
int main ()
{
static_assert(has_subscript< std::vector<int> >::value == true , "!");
static_assert(has_subscript< std::vector<double> >::value == true , "!");
static_assert(has_subscript< A >::value == true , "!");
static_assert(has_subscript< A, std::string >::value == false, "!");
static_assert(has_subscript< B >::value == false, "!");
static_assert(has_subscript< double[5] >::value == true , "!");
static_assert(has_subscript< double* >::value == true , "!");
static_assert(has_subscript< double >::value == false, "!");
}