新闻中心

c++怎么使用C++20的Concepts_c++20概念系统(Concepts)的语法与应用

2025-11-01
浏览次数:
返回列表
C++20的Concepts通过定义类型约束提升模板编程的可读性和错误提示清晰度。使用concept关键字和requires表达式可创建自定义约束,如Addable要求支持加法操作;标准库提供std::integral等常用Concept;可用于函数和类模板参数限制,并支持基于约束的重载,使编译错误更明确,代码更易维护。

c++怎么使用c++20的concepts_c++20概念系统(concepts)的语法与应用

C++20 的 Concepts 是一个强大的泛型编程工具,它允许你在编译期对模板参数施加约束,从而提升代码的可读性、可维护性和错误提示的清晰度。过去我们写模板时,如果传入了不合适的类型,编译错误往往是一长串难以理解的模板展开信息。而 Concepts 能让这些错误变得直观明了。

什么是 Concepts?

Concepts 是一种用于限制模板参数类型的机制。你可以把它看作“类型的要求清单”——只有满足这些要求的类型才能被用作模板实参。

例如,你希望某个函数模板只接受支持加法操作的类型(比如 int、double),就可以定义一个 Concept 来表达这个条件。

基本语法:如何定义和使用 Concept

定义 Concept 使用 concept 关键字,后接名称、等号以及一个布尔表达式。

示例:定义一个支持加法的类型约束

template<typename T>
concept Addable = requires(T a, T b) {
    a + b; // 检查是否能执行 a + b
};

然后在模板中使用:

template<Addable T>
T add(T a, T b) {
    return a + b;
}

现在调用 add(1, 2) 没问题,但如果你传入一个不支持 + 的类或指针类型,编译器会明确告诉你:“该类型不满足 Addable Concept”。

requires 表达式的几种形式

requires 是构建 Concept 的核心,它可以检查多种语义:

  • 简单要求(Simple requirement):只要表达式合法即可
requires(T t) { t++; }
  • 复合要求(Compound requirement):用花括号包裹,并可指定 noexcept 和返回类型
requires(T t) { { t++ } noexcept -> std::same_as<T>; }
  • 类型要求(Type requirement):检查某个类型是否存在
template<typename T>
concept HasValueType = requires {
    typename T::value_type; // 要求 T 有嵌套类型 value_type
};
  • 常量要求(Constant requirement):直接判断一个布尔值
concept Even = (4 % 2 == 0); // 总为 true,仅作演示

在函数模板中的应用

除了上面的 <addable t></addable> 写法,还有其他方式使用 Concept:

  • 使用 requires 子句
template<typename T>
T multiply(T a, T b) requires requires(T t){ t * t; }
{
    return a * b;
}
  • 结合 auto 使用(C++20 简化写法)
auto add(Addable auto a, Addable auto b) {
    return a + b;
}

这等价于:

短影AI 短影AI

长视频一键生成精彩短视频

短影AI 170 查看详情 短影AI
template<Addable T, Addable U>
auto add(T a, U b) { ... }

标准库中的常用 Concept

C++20 标准库在 <concepts></concepts> 头文件中提供了许多预定义的 Concept,可以直接使用:

  • std::integral:整型类型(int, char, bool 等)
  • std::floating_point:浮点类型(float, double)
  • std::default_constructible:可默认构造
  • std::copyable:可复制
  • std::equality_comparable:支持 == 和 !=

例子:只接受整数的函数

#include <concepts>
<p>void process(std::integral auto value) {
// 只能传入整型
}

在类模板中使用 Concept

也可以用来约束类模板参数:

template<typename T>
requires std::integral<T>
class Wrapper {
    T data;
public:
    Wrapper(T d) : data(d) {}
};

或者更简洁地:

template<std::integral T>
class Wrapper { ... };

优势与实际意义

  • 更好的编译错误信息:不再是一堆模板推导失败的堆栈,而是“你的类型不满足 XXX Concept”
  • 提高接口清晰度:从函数签名就能看出对类型的期望
  • 支持重载基于 Concept:可以根据不同的 Concept 提供多个函数版本

示例:根据 Concept 重载函数

void print(auto x) {
    std::cout << "任意类型: " << x << '\n';
}
<p>void print(std::integral auto x) {
std::cout << "整数: " << x << '\n';
}</p><p>void print(std::floating_point auto x) {
std::cout << "浮点数: " << x << '\n';
}

调用 print(42) 会匹配整数版本,print(3.14) 匹配浮点版本。

基本上就这些。Concepts 让模板编程从“靠运气通过编译”变成“有明确契约的类型系统”,是现代 C++ 泛型编程的重要进步。刚开始可能觉得 requires 写法有点复杂,但一旦掌握,你会觉得没有 Concepts 的模板很难再回头用了。

以上就是c++++怎么使用C++20的Concepts_c++20概念系统(Concepts)的语法与应用的详细内容,更多请关注其它相关文章!


# 错误提示  # 抚州营销推广价钱高吗  # 佛山网站建设 凤软网络  # 黄冈抖音seo公司排名  # 常州网站优化哪家合适  # 象山外贸推广seo  # 律师事务所营销推广方式  # 梁平企业网站建设案例  # 建立一个网站怎样推广  # 长沙销售型网站建设  # 郑州制作网站建设素材  # 是一个  # 客户端  # 网络编程  # c++20  # 从零开始  # 数据交换  # 不满足  # 浮点  # 整型  # 如何实现  # 标准库  # 编译错误  # c++  #   # 工具  # app  # concepts 


相关栏目: 【 科技资讯46185 】 【 网络学院92790


相关推荐: steam官方入口大全 steam账号注册及操作指南  Go语言JSON解析深度指南:动态访问与结构体映射实践  圆通快递查询实时追踪 圆通物流包裹状态快速查看  AO3官方在线访问地址 Archive of Our Own最新镜像合集  12306选座系统怎么选连座_12306选座多人连坐操作方法  高德地图总提示网络异常怎么办 高德地图离线导航设置与网络排查方法  windows10怎么查看本机ip_windows10命令提示符ipconfig使用  Promise错误处理:在catch后终止链式then执行的策略  快手赚钱渠道_快手收益来源  qq游戏大厅官方下载_qq游戏免费下载安装入口  Selenium Python中处理点击后新窗口加载冻结问题的策略与实践  Golang指针如何与map组合使用_Golang map指针组合实践  J*aScript生成器_j*ascript异步迭代  Python模块化编程:有效管理依赖与避免循环引用  优化LangChain文档加载与ChromaDB集成:解决多文档处理与分块问题  不同用户不同价格! 索尼开启账户个性化定价测试  腾讯视频怎么使用多账号家庭管理_腾讯视频家庭多账号统一管理与权限分配教程  Yandex免登录网页版地址 Yandex搜索引擎官方访问入口  在React函数组件中利用原生HTML5进行邮箱地址验证  不会效仿卡普空!《铁拳》制作人澄清:不采取赛事付费|直播|  “在文档元素之后找到了标记”是什么错误? 检查并修复XML中多个根元素的3个方法  CSS布局:解决全屏元素100%尺寸与外边距导致的页面溢出问题  cad如何更改注释性对象的比例_cad注释性比例调整方法  CSS子选择器:如何区分并样式化嵌套列表的子层级  lar*el怎么安全地存储和获取配置文件中的敏感信息_lar*el敏感信息安全存储方法  J*aScript中赋值与自增运算符的复杂交互与执行机制  Archive of Our Own官网直达 AO3最新可用地址一览  快速CSGO开箱网站指南 CSGO开箱平台推荐  React列表渲染与独立状态管理:避免全局状态影响局部更新  实现全屏滚动与导航点:专业教程  b站如何看历史记录_b站观看历史找回方法  CSS如何设置hover状态颜色_hover伪类调整背景或文字颜色  期待已久:小米17 Ultra、小米首款NAS本月登场  C++ map遍历方法大全_C++ map迭代器使用总结  Node.js CSV 数据处理:基于字段值条件过滤整条记录的策略  Basecamp怎样用留言钉固定重点_Basecamp用留言钉固定重点【重点标记】  Win11怎么关闭触摸屏_Windows 11禁用HID符合标准触摸屏  ExcelARRAYTOTEXT函数怎么自定义分隔符输出数组文本_ARRAYTOTEXT实现动态生成SQL语句  Pandas DataFrame:高效添加条件计算列  谷歌浏览器如何快速清除某个网站的数据_Chrome网站缓存清理方法  msn官网入口地址手机版 msn官方网站手机最新链接  抖音未来赚钱的新趋势 2025年值得关注的变现风口分析  Win11截图该按哪些键 Win11截屏完整流程解析【教程】  苹果手机如何防止被恶意App追踪  蛙漫2台版漫画地址 Manwa2正版网页版链接  如何将HTML表格多行数据保存到Google Sheets  优化MinIO list_objects_v2 操作的性能瓶颈与最佳实践  LINUX的I/O重定向是什么_深入理解LINUX中 >、>> 与 < 的区别  学习通网页版官方登录 超星学习通电脑端入口指南  解决Django多数据库/多Schema环境下外键迁移问题 

搜索