Clang 3.5:使用 SFINAE 检测不存在的全局函数
Clang 3.5: Detecting a gobal function doesn't exist using SFINAE
所以我有下面一段代码,在 VS2012 下可以完美运行。它检测全局函数 'Bar' 是否存在。
遗憾的是,这无法在 clang 3.5 下编译并出现此错误:
error : use of undeclared identifier 'Bar'
template <typename T> static int Foo( decltype( Bar )* ) { return 2; }
^
clang 支持这种东西吗?如果支持,正确的语法是什么?
谢谢
template <typename T> static int Foo( decltype( Bar )* ) { return 2; }
template <typename T> static int Foo(...) { return 1; }
void main()
{
printf("%d", Foo<int>(nullptr));
}
如果您知道要查找的函数的确切签名,可以使用 SFINAE 解决(使用 clang 3.6 和 gcc 4.9 测试):
#include <stdio.h>
int SomeMethod(double d) {
// nop
return 0;
}
struct HasSomeMethod {
typedef char no[2];
template <typename C>
static auto test(C c) -> decltype(SomeMethod(c));
//static auto test(C c) -> decltype(SomeOtherMethod(c));
template <typename>
static no& test(...);
static const bool value = sizeof(test<double>(1.0)) == sizeof(int);
};
int main()
{
printf("%d\n", HasSomeMethod::value);
}
如果将测试方法切换为注释版本,则输出 1 和 0。
它适用于任何全局方法,只要它至少有一个可以提取到模板参数的参数。
SFINAE 是 "Substitution Failure Is Not An Error" 的缩写。这意味着,您必须:
- 使用将由编译器替换的模板参数
- 您的主题必须以某种方式依赖于此模板参数
我的示例通过向主题函数提供模板化参数来解决后者 - 如果没有这样的函数,则无法替换,这是有效的。
如果没有您要查找的此类过载,结果相同。你可以在代码中,或者在测试点更改SomeMethod的参数,你就会看到。
不幸的是,对于这种情况,使用 type_traits
和类似的技巧是不可能的。例如下面的方法:
template<typename C>
static auto test(C c) -> typename std::is_same<decltype(SomeXXMethod()), decltype(c)>::type
和你原来的代码有同样的问题。对于成员检测,这是有效的,因为 C::SomeMethod
取决于模板参数。
由于这个要求,一个没有参数但非 void return 类型的方法可能可以被检查,但我现在不知道如何检查。
但是如果你有一个没有参数的 void 方法,我敢肯定你不会找到一个标准的方法来做到这一点 - 你找不到任何有效的类型规范同时包含你的类型和那个方法.
Clang 特定
如果您只想要 Clang 特定检查,有 Clang specific extensions
例如,您可以使用 #if __is_identifier(SomeMethod)
宏来检查给定的标识符是否存在。它可能是一个变量,或者一个函数,或者任何东西,但是在这个块中,你可以使用 SFINAE 来决定它的类型。
所以我有下面一段代码,在 VS2012 下可以完美运行。它检测全局函数 'Bar' 是否存在。
遗憾的是,这无法在 clang 3.5 下编译并出现此错误:
error : use of undeclared identifier 'Bar'
template <typename T> static int Foo( decltype( Bar )* ) { return 2; }
^
clang 支持这种东西吗?如果支持,正确的语法是什么?
谢谢
template <typename T> static int Foo( decltype( Bar )* ) { return 2; }
template <typename T> static int Foo(...) { return 1; }
void main()
{
printf("%d", Foo<int>(nullptr));
}
如果您知道要查找的函数的确切签名,可以使用 SFINAE 解决(使用 clang 3.6 和 gcc 4.9 测试):
#include <stdio.h>
int SomeMethod(double d) {
// nop
return 0;
}
struct HasSomeMethod {
typedef char no[2];
template <typename C>
static auto test(C c) -> decltype(SomeMethod(c));
//static auto test(C c) -> decltype(SomeOtherMethod(c));
template <typename>
static no& test(...);
static const bool value = sizeof(test<double>(1.0)) == sizeof(int);
};
int main()
{
printf("%d\n", HasSomeMethod::value);
}
如果将测试方法切换为注释版本,则输出 1 和 0。
它适用于任何全局方法,只要它至少有一个可以提取到模板参数的参数。
SFINAE 是 "Substitution Failure Is Not An Error" 的缩写。这意味着,您必须:
- 使用将由编译器替换的模板参数
- 您的主题必须以某种方式依赖于此模板参数
我的示例通过向主题函数提供模板化参数来解决后者 - 如果没有这样的函数,则无法替换,这是有效的。
如果没有您要查找的此类过载,结果相同。你可以在代码中,或者在测试点更改SomeMethod的参数,你就会看到。
不幸的是,对于这种情况,使用 type_traits
和类似的技巧是不可能的。例如下面的方法:
template<typename C>
static auto test(C c) -> typename std::is_same<decltype(SomeXXMethod()), decltype(c)>::type
和你原来的代码有同样的问题。对于成员检测,这是有效的,因为 C::SomeMethod
取决于模板参数。
由于这个要求,一个没有参数但非 void return 类型的方法可能可以被检查,但我现在不知道如何检查。
但是如果你有一个没有参数的 void 方法,我敢肯定你不会找到一个标准的方法来做到这一点 - 你找不到任何有效的类型规范同时包含你的类型和那个方法.
Clang 特定
如果您只想要 Clang 特定检查,有 Clang specific extensions
例如,您可以使用 #if __is_identifier(SomeMethod)
宏来检查给定的标识符是否存在。它可能是一个变量,或者一个函数,或者任何东西,但是在这个块中,你可以使用 SFINAE 来决定它的类型。