0

0

c++20的Concepts如何彻底替代SFINAE? (语法对比)

尼克

尼克

发布时间:2026-01-21 18:05:02

|

739人浏览过

|

来源于php中文网

原创

Concepts 不能彻底替代 SFINAE,二者机制不同:Concepts 在参数替换后、重载解析前检查约束并直接剔除不满足者,错误更清晰;SFINAE 在替换过程中静默丢弃失败特化,仍参与重载解析,适用于动态、异常、元编程等场景。

c++20的concepts如何彻底替代sfinae? (语法对比)

Concepts 不能“彻底替代” SFINAE,它只是让大部分原本需要 SFINAE 的场景变得更清晰、更安全、更易读;SFINAE 在底层约束失败时仍会参与重载解析,而 Concepts 是在更早的约束检查阶段就排除不满足条件的候选,两者机制不同,目标也不完全重合。

Concepts 和 SFINAE 的触发时机完全不同

Concepts 是在模板参数替换完成之后、重载决议(overload resolution)开始之前进行约束检查;一旦 requires 子句不满足,该模板特化直接被从候选集中移除,不进入重载集。SFINAE 则依赖于“替换失败不是错误”的规则,在模板参数替换过程中,若某表达式(如 decltype(T{}.size()))因类型不支持而无法推导,该特化仅被静默丢弃——但这个过程发生在更底层、更不可控的位置。

这意味着:

  • Concepts 错误信息更短、更聚焦于约束本身(比如 "T does not satisfy std::regular"),而 SFINAE 错误常堆叠十几层 decltype / std::enable_if 展开,难以定位
  • Concepts 不会因“意外替换失败”导致重载歧义或隐式转换干扰;SFINAE 容易因过度泛化(如漏写 std::decay_t)让本不该参与重载的模板意外存活
  • Concepts 无法表达“仅当某表达式可求值时才启用”的动态条件(例如:只在 T 支持 operator+ 且结果可转为 int 时启用),这种仍需 SFINAE 或 requires 中嵌套 decltype 检查

用 requires 替代 enable_if 的典型写法对比

下面是一个判断是否支持 begin()/end() 迭代器访问的函数模板,分别用 SFINAE 和 Concepts 实现:

立即学习C++免费学习笔记(深入)”;

SFINAE 版本(C++11/14 风格):

慧中标AI标书
慧中标AI标书

慧中标AI标书是一款AI智能辅助写标书工具。

下载
template
auto range_size(const T& t) -> decltype(t.end() - t.begin(), std::size_t{}) {
    return t.end() - t.begin();
}

Concepts 版本(C++20):

template
    requires requires(const T& t) { t.begin(); t.end(); }
auto range_size(const T& t) {
    return std::distance(t.begin(), t.end());
}

关键差异点:

  • requires 块内是纯表达式检查,不涉及类型推导或 decltype 嵌套;失败时直接报“T does not satisfy the requires clause”
  • SFINAE 版本中,decltype(t.end() - t.begin(), std::size_t{}) 的逗号表达式容易误写成 decltype(t.end() - t.begin()),导致对非随机访问迭代器(如 std::list)编译失败而非静默禁用
  • Concepts 版本可复用为命名 concept:template concept range = requires(const T& t) { t.begin(); t.end(); };,后续所有函数都能直接写 range T

Concepts 无法覆盖的 SFINAE 场景

以下情况目前仍需 SFINAE(或 Concepts + SFINAE 混合):

  • 需要根据某个表达式是否“可求值且不抛异常”做重载选择(noexcept 检查),Concepts 的 requires 不检查异常规范
  • 想让一个函数模板对 std::vector 启用特化 A,对 std::vector 启用特化 B,但两者都满足同一 concept;此时必须靠 SFINAE 或部分特化区分
  • 实现类似 std::is_convertible_v 这类元函数,其本质就是靠 SFINAE 在 sfinae-friendly context 中尝试构造,Concepts 无法替代这种“试探性编译”逻辑

真正要放弃 SFINAE,得先确认你的约束全是静态、正向、类型层面的——比如“支持 operator”“有嵌套 value_type”“可默认构造”。一旦涉及运行时语义、异常、隐式转换路径或元编程探测,SFINAE 或 std::void_t 技巧依然不可绕过。

相关专题

更多
c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

525

2023.09.20

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

338

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

542

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

53

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

53

2025.08.29

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

99

2025.10.23

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

392

2023.07.18

Java编译相关教程合集
Java编译相关教程合集

本专题整合了Java编译相关教程,阅读专题下面的文章了解更多详细内容。

9

2026.01.21

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 7.2万人学习

C 教程
C 教程

共75课时 | 4.1万人学习

C++教程
C++教程

共115课时 | 13.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号