首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

C++那些事之SFINAE

介绍c++的SFINAE概念:类成员的编译时内省 0.导语1.C++自省?...根据名称找出所有适用的函数和函数模板对于适用的函数模板,要根据实际情况对模板形参进行替换; 替换过程中如果发生错误,这个模板会被丢弃 在上面两步生成的可行函数集合中,编译器会寻找一个最佳匹配,产生对该函数的调用...2.2 SFINAE 回忆一下上述的重载决议: 函数调用 函数模板 SFINAE 我已经用几个段落的强大功能来戏弄你了,现在终于可以解释这个并不复杂的缩写词了。...SFINAE表示替换失败不是错误( Substitution Failure Is Not An Error)。简单地说,替换就是尝试用提供的类型或值替换模板参数的机制。...您可以使用此解决方案的变体对类型进行大量测试(测试成员,子类型...),我建议您更多地搜索SFINAE技巧。

2.2K20

现代C++之SFINAE

介绍c++的SFINAE概念:类成员的编译时内省 0.导语1.C++自省?...根据名称找出所有适用的函数和函数模板对于适用的函数模板,要根据实际情况对模板形参进行替换; 替换过程中如果发生错误,这个模板会被丢弃 在上面两步生成的可行函数集合中,编译器会寻找一个最佳匹配,产生对该函数的调用...2.2 SFINAE 回忆一下上述的重载决议: 函数调用 函数模板 SFINAE 我已经用几个段落的强大功能来戏弄你了,现在终于可以解释这个并不复杂的缩写词了。...SFINAE表示替换失败不是错误( Substitution Failure Is Not An Error)。简单地说,替换就是尝试用提供的类型或值替换模板参数的机制。...您可以使用此解决方案的变体对类型进行大量测试(测试成员,子类型...),我建议您更多地搜索SFINAE技巧。

3K20
  • 您找到你想要的搜索结果了吗?
    是的
    没有找到

    C++模板编程:深入理解分离编译的挑战与解决方案

    然后,我们将详细介绍几种常用的模板分离编译方法,包括显式实例化声明、包含模型、预编译头文件和模板库等。通过这些方法,我们可以有效地管理模板的分离编译问题,确保在多个翻译单元中正确地实例化和使用模板。...在C++模板中,特别是当模板参数依赖于模板本身时,编译器有时可能无法区分一个名称是指代类型还是对象。在这种情况下,使用typename关键字可以显式地告诉编译器该名称是一个类型。...特化不能改变模板的接口:特化版本必须提供与一般化版本相同的成员函数和接口,否则会导致编译错误。 特化不能增加新的模板参数:特化版本不能增加新的模板参数,它必须匹配一般化版本中的参数数量。...对于非指针类型,将使用泛型版本的Less函数。 3.2 使用SFINAE模拟函数模板的特化 SFINAE是一种强大的技术,它允许我们在模板编程中根据类型特征来选择性地启用或禁用模板的某些实例化。...C++不支持函数模板的偏特化。 可以使用SFINAE技术来模拟函数模板的特化行为,但这通常涉及到条件编译和模板的实例化选择。

    20110

    未来已来:从SFINAE到concepts

    ,将参数x赋值给一个string类型的v,但是在main()函数中 ,调用fun()函数时候传入了1,这个编译器会推导为int类型,那么把一个int类型赋值给string,编译器会报错。...这是一种 C++ 中的编译期技术,用于在模板实例化过程中,当尝试进行模板参数的替换时,如果出现了替换失败(通常是由于找不到相应的成员函数、操作符等),不会导致编译错误,而是会选择其他可行的模板特化。...它的核心思想是,如果在模板参数的替换中遇到了错误,编译器不应该报错,而是应该简单地将这个特化从候选列表中移除。这样,即使部分模板特化失败,编译仍然可以继续进行,选择其他可行的特化。...这一机制使得在模板元编程中能够更加灵活地根据类型的特性选择不同的实现路径。...成员函数 如果要判断某个类是否存在某个成员函数,那么可以像如下这么写: #include #include #include template

    25410

    【笔记】《深入理解C++11》(上)

    有了extern后编译器会自动删除重复的实例化模板, 不但节省内存还节省了多余的实例化时间 注意被其他文件调用的外部模板一定要在要用到的类实例化之前实例化 局部和匿名成员可以作为模板实参了, 但仍要注意匿名类型的声明不能在参数位置...没有继承关系) 类中的第一个非静态成员类型要与基类不同(为了类指针能直接指向第一个成员) 没有虚函数和虚基类 所有非静态成员都满足POD布局(递归定义) 之所以C++11引入POD的概念是为了保证我们可以安全地用...Substitution_failure_is_not_an_error SFINEA: Substitution failure is not an error, 替换失败不是错误 这个词是在标准化地描述如何为函数模板进行合理的参数匹配...函数模板是根据我们的实参类型在调用时进行特化并实例化的, 具体来说匹配遵循以下步骤: 首先对于一次调用, 编译器查找所有具有此名称的函数和实例化的模板函数表 在这些函数中进行比较, 将不可行的函数剔除,...// 由于SFINEA的原因int对f1的尝试不算做实例化的error, 而属于匹配过程中的一次failure f(10); } 基于对这个匹配过程的标准化描述, 我们可以无关编译器地对模板匹配的可行性进行判断

    2K20

    【C++篇】领略模板编程的进阶之美:参数巧思与编译的智慧

    这是因为模板的实例化是由编译器根据实际使用的类型生成的代码,如果在模板的定义和使用之间缺乏可见性,编译器无法正确地实例化模板。...Add(1.0, 2.0); // 使用模板函数 return 0; } 在这种情况下,由于模板的定义和使用是分离的,编译器在不同编译单元中无法找到模板的定义,从而导致链接错误。...7.1.1 优先调用非模板函数 在匹配时,编译器会优先选择非模板函数,如果有完全匹配的非模板函数存在,编译器会选择该函数,而不是实例化模板。...return 0; } 7.1.2 如果没有非模板函数,匹配模板实例 如果没有完全匹配的非模板函数存在,编译器将生成模板实例化版本。...SFINAE 是指在模板实例化过程中,如果某些模板参数的替换失败,编译器不会直接报错,而是选择其他可行的模板。

    14010

    C++设计模式之SFINAE:用来检测类中是否有某个成员函数

    针对类中特定成员函数的检测其实在工作中也可能用到。C++中可以用SFINAE技巧达到这个目的。...SFINAE是Substitution Failure Is Not An Error的缩写,直译为:匹配失败不是错误。属于C++模板编程中的高级技巧,但属于模板元编程中的基本技巧。...在这个过程中,我发现有些常见的SFINAE写法是有问题的,下面探讨一下。 举个例子,我们来check一下C++标准库的类中有没有push_back()成员函数。...如果是检测其他成员函数,比如size则不需要这么麻烦只要一个Helper即可。 而test函数,对于返回true的模板函数,其参数是一个指针类型。...如果需求是要检测任意成员函数,而不限定是哪个函数的话,毫无疑问,需要借助宏了。将上面的代码改变成宏的版本,push_back作为宏的一个参数,即可。 我这里为什么用push_back()举例呢?

    4.6K20

    C++奇淫巧技之SFINAE

    SFINAE 技术,即匹配失败不是错误,英文Substitution Failure Is Not An Error,其作用是当我们在进行模板特化的时候,会去选择那个正确的模板,避免失败 看个具体的例子...multiplication_result multiply(T t1, T t2) { return t1 * t2; } int main(void) { multiply(4,5); } 当我们编译的时候...,会去匹配模板 multiply,但是由于我们不知道multiplication_result,根据 Substitution Failure Is Not An Error ,于是我们就去选择函数 multiply...这种技术在代码中的一个大的用途就是在编译时期来确定某个 type 是否具有我们需要的性质,看代码 template struct is_pointer { template...,3个是接受不同的指针参数,另一个则包括了其他的所有参数, IntPtr 是一个变量指针 FooMemberPtr 是一个成员属性指针 FuncPtr 是一个函数指针 接着我们来看下 muduo

    53330

    C++ 模板沉思录(上)

    2.3 SFINAE SFINAE(Substitution Failure Is Not An Error,替换失败并非错误)是一个高级模板技巧。...(这样我们似乎就能让编译器在“喜欢和不喜欢”之间做出抉择) 也许你将恍然大悟:类的成员指针。...显然,当编译器拿到一个T,并准备生成一个“T::*”时,仅当T是一个类类型时,这一生成才是正确的,合乎语法的;否则,这个函数签名将根本无法被生成出来,从而进一步的使得编译器“被迫”选择那个“最不喜欢的版本...编译期分数类的实现非常简单,我们只需要通过一个“构造函数”将模板参数保留下来,作为静态数据成员即可。...而比大小的结果,就是“比大小模板”的一个数据成员。

    1.3K20

    浅谈 C++ 元编程

    转化为常量表达式,类似测试表达式实现重载的选择(但需要添加一个冗余的 函数参数/函数返回值/模板参数); std::void_t 直接 检查依赖 的成员/函数是否存在,不存在则无法重载(可以用于构造谓词...(two-phase name lookup)的规定,如果直接使用 static_assert (false) 断言,会在模板还没实例化的第一阶段无法通过编译)。...现代 C++ 也不断地增加语言的特性,致力于降低元编程的复杂性: C++ 11 的 别名模板提供了对模板中的类型的简记方法; C++ 14 的 变量模板提供了对模板中常量的简记方法; C++ 17 的 ...如果需要调试的是一段通过很多次的 编译时测试和 编译时迭代展开的代码,即这段代码是各个模板的拼接生成的(而且展开的层数很多);那么,调试时需要不断地在各个模板的 实例 (instance) 间来回切换。...但是,过度的使用元编程,一方面会 增加编译时间,另一方面会 降低程序的可读性。不过,在 C++ 不断地演化中,新的语言特性被不断提出,为元编程提供更多的可能。

    3.1K61

    【笔记】《深入理解C++11》(下)

    , 在表达式面前加上constexpr来声明 编译器会在编译时期对constexpr进行值计算, 类似模板元编程 constexpr可以在函数前声明, 但是C++11的时候常量表达式函数有很多限制, 后来的版本渐渐放松限制...注意此时类的构造函数函数体必须为空, 所有成员都只能依靠常量表达式在初始化列表中初始化 常量表达式不能用于virtual 常量表达式函数不需要重写非常量版本, 编译器会自动生成, 重写反而会报错 当模板函数声明为常量表达式后..., 如果函数的实例化结果不满足常量表达式要求, 那么常量表达式符号会被忽略而不会报错(也是一种SFINAE) 变长模板 C标准中的变长宏不强调类型并不安全 C++11中的tuple模板就是典型的变长模板...注意函数参数包必须是函数的最后一个参数包(模板参数包没有这个要求) 模板参数包可以在以下七个位置展开, 但是不用太关注, 这些展开位置基本覆盖常见需求: 表达式 初始化列表 基类描述列表 类成员初始化列表...很直观的, 类型不完整的类无法通过alignof的编译 // 这里的Color按照8字节对齐, 但是结构体显然大多数时侯是一整个一整个地访问的 // 那么8字节的对齐并不能提高读写的效率, 也不能充分利用缓存

    1.1K31

    C++20初体验——concepts

    然而_RandomAccessIterator毕竟只是一个名字,编译器不知道它表示哪些要求,更无法据此输出错误信息。...这些是句法上无法检查的,所以这两个concept更像是一种规约:如果模板参数被这种concept约束,那么客户调用时传入的参数就得满足这些语义需求。...如果模板参数代入时出现了不存在的类型或变量,该约束仅仅是不被满足,而不会产生编译错误。 约束可以用于函数模板、类模板和成员函数,非模板类的非模板成员函数除外。...函数模板与类模板的约束是类似的,只有满足约束时模板才能实例化;对于成员函数的约束,如果它作用于模板类的模板参数,当约束不满足时,并不是类模板不能被实例化,而是实例化后的模板类没有这个成员函数: #include...);唯独第三条没有解决,导致冗长的模板错误,并且衍生出以SFINAE为代表的一些奇技淫巧。

    1.4K10

    C++泛型编程泛泛谈

    这里可能要插一个东西叫,元编程: 模板元编程把模板的一些技术(特化、实例化、 SFINAE )当成模板元编程这种特定语言的控制流。...**注:**在模板定义中,模板参数列表不能为空 模板参数列表表示在类或函数定义中用到的类型或者值。当我们使用模板的时候,可以(显式或隐式地)指定模板实参,将其绑定到模板参数上。...通常来说,我们将类定义和函数说明放在头文件中,而普通函数和类的成员函数的定义放在源文件中,模板则不尽相同:为了生成一个实例化的版本,编译器需要掌握函数模板或类模板成员函数的定义。...总结:与非模板代码不同,模板的头文件通常既包括声明也包括定义,即函数模板和类模板成员函数的定义通常放在头文件中。...类模板成员函数的实例化 默认的情况下,一个类模板的成员函数只有在程序用到它的时候才会实例化。 函数重载与模板特例化的区别 当定义函数模板的特例化版本时,我们本质上接管了编译器的工作。

    1K30

    C++一分钟之-模板基础:泛型编程

    本文将深入浅出地介绍C++模板的基础概念、常见问题、易错点以及如何有效避免这些问题,并通过具体的代码示例加以说明。...模板基础:一石多鸟的利器 概念 模板允许你定义一个函数或类,在其中指定一种或多种类型参数。编译器会根据实际调用时提供的类型参数生成具体的代码。...常见问题与易错点 模板特化与偏特化混淆:模板特化用于完全指定所有模板参数,而偏特化则是部分指定。错误地使用会导致编译错误或意料之外的行为。...利用现代C++特性:如SFINAE(Substitution Failure Is Not An Error)和std::enable_if等,优雅地处理模板元编程中的条件编译。...通过了解上述常见问题和避免策略,结合实际编程练习,你将能更加得心应手地运用模板,编写出既高效又优雅的C++代码。

    11010

    C++模版的本质

    : 函数模板的签名包括模板参数,返回值,函数名,函数参数, cv-qualifier; 函数模板编译顺序大致:名称查找(可能涉及参数依赖查找)->实参推导->模板实参替换(实例化,可能涉及 SFINAE...)->函数重载决议->编译; 函数模板可以在实例化时候进行参数推导,必须知道每个模板的实参,但不必指定每个模板的实参。...编译器会从函数实参推导缺失的模板实参。...); 函数模板实例化过程中,参数推导不匹配所有的模板或者同时存在多个模板实例满足,或者函数重载决议有歧义等,实例化失败; 为了编译函数模板调用,编译器必须在非模板重载、模板重载和模板重载的特化间决定一个无歧义最佳的模板...SFINAE -Substitution failure is not an error 要理解这句话的关键点是failure和error在模板实例化中意义,模板实例化时候,编译器会用模板实参或者通过模板实参推导出参数类型带入可能的模板集

    1.7K30

    你经历过哪些优秀的C++面试?

    深入问题:在设计大型系统时,你如何避免由于过度使用虚函数导致的性能问题?如何在需要高性能的地方绕开虚函数? 4、C++ 标准库与模板元编程 问题:解释模板的偏特化和全特化。...对 std::enable_if、SFINAE (Substitution Failure Is Not An Error) 等高级模板技术的掌握。...深入问题:请实现一个基于模板元编程的类型推导系统,能够在编译期推导出一个函数返回的类型,并结合 SFINAE 做出函数的选择。 5、性能优化与代码设计 问题:给定一段代码,分析其性能瓶颈。...7、编译器原理与底层实现 问题:解释 C++ 编译过程中的各个阶段:预处理、编译、汇编、链接。编译器是如何将模板代码实例化为具体实现的?...深入问题:编译优化中的 inline、constexpr 和模板展开有何不同?这些优化在不同场景下如何影响性能?

    13610
    领券