替换失败并非错误 - 维基百科,自由的百科全书

替换失败并非错误 (Substitution failure is not an error, SFINAE)是指C++语言在模板参数匹配失败时不认为这是一个编译错误。戴维·范德沃德英语David Vandevoorde最先引入SFINAE缩写描述相关编程技术。[1]

具体说,当创建一个重载函数的候选集时,某些(或全部)候选函数是用模板实参替换(可能的推导)模板形参的模板实例化结果。如果某个模板的实参替换时失败,编译器将在候选集中删除该模板,而不是当作一个编译错误从而中断编译过程,这需要C++语言标准授予如此处理的许可。[2] 如果一个或多个候选保留下来,那么函数重载的解析就是成功的,函数调用也是良好的。

例子

[编辑]

下属简单例子解释了SFINAE:

struct Test {   typedef int foo; };  template <typename T> void f(typename T::foo) {}  // Definition #1  template <typename T> void f(T) {}  // Definition #2  int main() {   f<Test>(10);  // Call #1.   f<int>(10);   // Call #2. 并无编译错误(即使没有 int::foo)                 // thanks to SFINAE. } 

在限定名字解析时(T::foo)使用非类的数据类型,导致f<int>推导失败因为int并无嵌套数据类型foo, 但程序仍是良好定义的,因为候选函数集中还有一个有效的函数。

虽然SFINAE最初引入时是用于避免在不相关模板声明可见时(如通过包含头文件)产生不良程序。许多程序员后来发现这种行为可用于编译时内省(introspection)。具体说,在模板实例化时允许模板确定模板参数的特定性质。

例如,SFINAE用于确定一个类型是否包含特定typedef:

#include <iostream>  template <typename T> struct has_typedef_foobar {   // Types "yes" and "no" are guaranteed to have different sizes,   // specifically sizeof(yes) == 1 and sizeof(no) == 2.   typedef char yes[1];   typedef char no[2];    template <typename C>   static yes& test(typename C::foobar*);    template <typename>   static no& test(...);    // If the "sizeof" of the result of calling test<T>(nullptr) is equal to   // sizeof(yes), the first overload worked and T has a nested type named   // foobar.   static const bool value = sizeof(test<T>(nullptr)) == sizeof(yes); };  struct foo {   typedef float foobar; };  int main() {   std::cout << std::boolalpha;   std::cout << has_typedef_foobar<int>::value << std::endl;  // Prints false   std::cout << has_typedef_foobar<foo>::value << std::endl;  // Prints true } 

当类型T有嵌套类型foobartest的第一个定义被实例化并且空指针常量被作为参数传入。(结果类型是yes。)如果不能匹配嵌套类型foobar,唯一可用函数是第二个test定义,且表达式的结果类型为no。省略号(ellipsis)不仅用于接收任何类型,它的转换的优先级是最低的,因而优先匹配第一个定义,这去除了二义性。

C++11的简化

[编辑]

C++11中,上述代码可以简化为:

#include <iostream> #include <type_traits>  template <typename... Ts> using void_t = void;  template <typename T, typename = void> struct has_typedef_foobar : std::false_type {};  template <typename T> struct has_typedef_foobar<T, void_t<typename T::foobar>> : std::true_type {};  struct foo {   using foobar = float; };  int main() {   std::cout << std::boolalpha;   std::cout << has_typedef_foobar<int>::value << std::endl;   std::cout << has_typedef_foobar<foo>::value << std::endl; } 

C++标准的未来版本中Library fundamental v2 (n4562)页面存档备份,存于互联网档案馆)建议把上述代码改写为:

#include <iostream> #include <type_traits>  template <typename T> using has_typedef_foobar_t = typename T::foobar;  struct foo {   using foobar = float; };  int main() {   std::cout << std::boolalpha;   std::cout << std::is_detected<has_typedef_foobar_t, int>::value << std::endl;   std::cout << std::is_detected<has_typedef_foobar_t, foo>::value << std::endl; } 

Boost的使用者在boost::enable_if[3]中使用SFINAE。

参考文献

[编辑]
  1. ^ Vandevoorde, David; Nicolai M. Josuttis. C++ Templates: The Complete Guide. Addison-Wesley Professional. 2002. ISBN 0-201-73484-2. 
  2. ^ International Organization for Standardization. "ISO/IEC 14882:2003, Programming languages — C++", § 14.8.2.
  3. ^ Boost Enable If. [2020-08-18]. (原始内容存档于2008-09-05).