0

0

C++ 概念(Concepts)是什么?(如何利用 C++20 约束模板参数)

穿越時空

穿越時空

发布时间:2026-02-13 15:09:02

|

809人浏览过

|

来源于php中文网

原创

concepts是编译期强制执行的类型约束机制,用于精准检查模板参数是否满足特定操作要求,提升错误定位精度与重载解析能力。

c++ 概念(concepts)是什么?(如何利用 c++20 约束模板参数)

Concepts 是模板参数的类型契约

它不是新语法糖,而是编译期强制执行的约束机制:你写 std::sortable,编译器真会检查传入类型是否支持 std::ranges::sort 所需的操作(比如可比较、可交换),而不是等实例化失败后甩出一页看不懂的错误。

常见错误现象是——没用 Concepts 时,模板函数对非法类型报错位置极深,堆栈里全是 __sort_impl__iter_swap 这类内部符号;用了 Concepts 后,错误直接定位到调用点:“static_assert failed: ‘T must satisfy std::sortable’”。

  • 使用场景集中在泛型库开发、算法封装、接口抽象层(比如你要写一个通用容器适配器,得确保用户传的迭代器满足 std::random_access_iterator
  • 不等于 type trait:前者是“能用”,后者是“是什么”;std::is_integral_v<t></t> 告诉你 T 是整数,std::integral 则要求 T 必须支持 +== 等一整套操作
  • 定义 Concepts 用 concept 关键字,但约束逻辑必须靠已有的 type trait 或 requires 表达式拼装,不能凭空声明语义

怎么写一个最小可用的自定义 Concept

别从复杂约束起步。先锁定一个明确操作,比如“能用 operator* 解引用并得到 int”:

template<typename T>
concept deref_to_int = requires(T t) {
    { *t } -> std::same_as<int>;
};

注意三点:

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

虎课网
虎课网

虎课网是超过1800万用户信赖的自学平台,拥有海量设计、绘画、摄影、办公软件、职业技能等优质的高清教程视频,用户可以根据行业和兴趣爱好,自主选择学习内容,每天免费学习一个...

下载
  • requires 块里写的是“表达式能通过编译 + 类型匹配”,不是运行时逻辑;{ *t } 检查能否解引用,-> std::same_as<int></int> 检查结果类型是不是 exactly int(不是 long,也不是可隐式转为 int 的类型)
  • 参数名 t 只是占位符,不参与求值,所以即使 T 是未定义类型或空类,也不会崩溃
  • 别在 concept 定义里调用具体函数(如 foo(t)),除非你已确保 foo 在该作用域可见且重载解析明确,否则容易触发 ODR-violation 或 SFINAE 意外失效

把 Concepts 用在函数模板上,不是加个关键词就完事

错误写法:template<typename t> void f(T x) requires std::integral<t></t></typename> —— 这只是让编译器多报一次 constraint failure,没解决重载歧义。

真正起效的方式是让 Concepts 参与重载决议:

void f(std::integral auto x) { /* 处理整数 */ }
void f(std::floating_point auto x) { /* 处理浮点 */ }

这时调用 f(42) 会精确匹配第一条,而 f(3.14) 匹配第二条。关键点:

  • concept-name auto 替代 typename T,才能启用基于 Concepts 的重载排序(concept 越严格,优先级越高)
  • 如果两个 concept 有包含关系(比如 std::signed_integralstd::integral 的子集),编译器会优先选更特化的那个
  • 混用传统 SFINAE 和 Concepts 很危险:比如 template<typename t std::enable_if_t>>* = nullptr></typename>void g(std::arithmetic auto) 共存,可能因约束强度判断不一致导致意外选择

为什么你的 Concepts 编译慢、报错还乱

根本原因是 constraint 检查本身要实例化大量辅助模板。比如 std::sortable 内部会展开 std::indirectly_swappablestd::indirect_strict_weak_order 等一串依赖,每层都做 requires 检查。

  • 避免在 requires 表达式里写复杂嵌套:比如 { some_template<t>::value }</t> 会触发 some_template 实例化,若它本身又依赖其他未约束类型,错误信息立刻爆炸
  • 调试时先关掉所有非必要 constraint:把 requires A && B && C 拆成三个独立函数,逐个验证哪条卡住
  • Clang 错误比 GCC 更友好,尤其对 requires 块内表达式失败的提示;但两者对 concept 继承链(concept D = A && B)的展开深度控制都不够好,超三层建议手动拆解

实际项目里,Concepts 不是用来给每个模板参数都套上金箍的,而是守住关键接口边界——比如容器的 value_type 必须可移动,算法的迭代器必须满足某类访问模式。约束太松,失去意义;约束太紧,连 std::vector<bool></bool> 这种特化都过不了。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
sort排序函数用法
sort排序函数用法

sort排序函数的用法:1、对列表进行排序,默认情况下,sort函数按升序排序,因此最终输出的结果是按从小到大的顺序排列的;2、对元组进行排序,默认情况下,sort函数按元素的大小进行排序,因此最终输出的结果是按从小到大的顺序排列的;3、对字典进行排序,由于字典是无序的,因此排序后的结果仍然是原来的字典,使用一个lambda表达式作为key参数的值,用于指定排序的依据。

399

2023.09.04

string转int
string转int

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

709

2023.08.02

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

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

561

2024.08.29

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

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

213

2025.08.29

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

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

206

2025.08.29

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

181

2023.11.23

java中void的含义
java中void的含义

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

114

2025.11.27

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1417

2023.10.19

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

15

2026.02.13

热门下载

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

精品课程

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

共94课时 | 9.3万人学习

C 教程
C 教程

共75课时 | 4.7万人学习

C++教程
C++教程

共115课时 | 17.5万人学习

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

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