0

0

c++20的requires子句和if constexpr如何协同工作? (模板约束)

穿越時空

穿越時空

发布时间:2026-01-21 17:07:12

|

263人浏览过

|

来源于php中文网

原创

requires 是编译期约束守门人,负责模板实例化初期筛选合法类型;if constexpr 是函数体内条件开关,对已满足约束的类型做细化分支。二者阶段不同、分工明确,不可互替。

c++20的requires子句和if constexpr如何协同工作? (模板约束)

requires 子句负责编译期约束,if constexpr 负责分支内细化逻辑

requires 是模板声明层面的“守门人”,它在模板实例化初期就检查概念是否满足;而 if constexpr 是函数体内的“条件开关”,只对已通过 requires 的类型做进一步分支判断。二者不在同一阶段工作,不能互相替代,但能形成清晰分工:前者筛掉非法类型,后者在合法类型中按细节做差异化实现。

典型组合写法:先用 requires 过滤,再用 if constexpr 分支

常见错误是试图在 requires 中写复杂逻辑(比如调用未定义的 SFINAE 表达式),或在 if constexpr 中处理本该由 requires 拦截的非法类型(导致硬错误)。正确做法是:

  • requires 只依赖已定义的概念(如 std::integral、自定义 Sortable)或简单表达式(如 T::value 是否可访问)
  • if constexpr 内部可安全使用 decltypesizeof、成员检测等,因为此时 T 已确定满足约束
  • 若需多级分类(比如先分整型/浮点,再分有符号/无符号),requires 做第一层粗筛,if constexpr 做第二层细判
template
requires std::integral || std::floating_point
T normalize(T x) {
    if constexpr (std::floating_point) {
        return x < 0 ? -x : x; // fabs 不必要,且 float/double 行为一致
    } else {
        return x < 0 ? -x : x; // 对 signed int 安全;unsigned 不会进此分支(被 requires 拦住)
    }
}

requires 失败时 if constexpr 不会被解析,但反之不成立

这是关键行为差异:requires 检查失败 → 整个模板不参与重载决议 → 函数体(含 if constexpr)根本不会被实例化;而 if constexpr 分支里若写了非法代码(比如对 T 调用不存在的 .size()),只要该分支未被选中,就不会报错 —— 但前提是该模板本身已通过 requires 或其他约束进入候选集。

  • 错误示例:给 std::string 类型调用 normalize<:string>requires 失败 → 编译器直接忽略该特化,连 if constexpr 都不看
  • 危险示例:删掉 requires,仅靠 if constexpr (has_size_v) 分支 → 若传入 intelse 分支里若误写 t.size() 就会触发硬错误
  • 所以 requires 是兜底,if constexpr 是优化,顺序不能颠倒

性能与可读性权衡:别把所有判断都塞进 if constexpr

过度依赖 if constexpr 会让单个函数体膨胀,尤其当分支逻辑复杂时。更清晰的做法是用 requires 拆成多个约束更精确的函数重载:

燕雀Logo
燕雀Logo

为用户提供LOGO免费设计在线生成服务

下载

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

template
requires std::signed_integral
T abs_impl(T x) { return x < 0 ? -x : x; }

template
requires std::unsigned_integral
T abs_impl(T x) { return x; } // 无符号类型无需计算

template
requires std::floating_point
T abs_impl(T x) { return std::abs(x); }

这种写法比一个带三层 if constexpr 的函数更易维护,编译器生成的代码也更干净。只有当分支逻辑高度相似、仅差一两个操作时,才值得用 if constexpr 合并。

真正容易被忽略的是:requires 约束一旦写错(比如漏掉 const 限定或引用类别),会导致重载决议意外失败;而 if constexpr 的条件表达式若依赖 ADL 或未完全定义的 trait,可能在某些编译器上表现不一致。务必用 static_assert 在函数体内补一手验证。

相关专题

更多
string转int
string转int

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

338

2023.08.02

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

757

2023.08.22

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

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

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

5

2026.01.21

C++多线程相关合集
C++多线程相关合集

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

0

2026.01.21

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

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

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