C++20中Concepts怎么定义_C++泛型编程模板约束使用教程【特性】

7次阅读

concepts是编译期断言工具,用于显式表达接口意图并提前报错;最简写法为templateconcept name=requires(t a,t b){{a+b}->std::same_as};

C++20中Concepts怎么定义_C++泛型编程模板约束使用教程【特性】

Concepts 不是“教程式语法糖”,而是编译期断言工具——它不改变模板逻辑,只让错误提前、让接口意图显式化。

怎么写一个最简 concept(c++20 要求)

必须用 concept 关键字 + 布尔常量表达式,不能是函数体或运行时逻辑:

template<typename T> concept Addable = requires(T a, T b) {     { a + b } -> std::same_as<T>; };
  • requires 后面是约束块,不是代码执行,仅做语法/语义可行性检查
  • 返回类型约束用 -> std::same_as<t></t>,不是 -> T(后者不检查值类别和 cv 限定)
  • 变量名(如 a, b)只是占位符,不参与求值;类型必须可推导,不能写 int a

为什么 template 比 enable_if 更直观

传统 std::enable_if 把约束藏在模板参数列表末尾,报错时深、信息散;而 concept 直接出现在形参位置,编译器能精准定位不满足的约束点:

// 错误提示直接说 "T does not satisfy Addable" void foo(Addable auto x) { /* ... */ } <p>// 而不是:error: no type named 'type' in 'struct std::enable_if<false, void>'

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

  • 使用 Addable auto 是最轻量写法,等价于 template<addable t> void foo(T x)</addable>
  • 若需复用类型名(比如在函数体内用 T),仍得写完整模板头,不能省略 template<addable t></addable>
  • 多个 concept 可用 &&|| 组合,但注意短路不生效(全部静态检查)

常见踩坑:requires 表达式里不能出现未定义行为

哪怕只是“看起来合法”的表达式,只要触发未定义行为(如解引用空指针、越界访问),整个 concept 就变成不满足(false),且无警告:

  • 禁止在 requires 中调用可能抛异常或有副作用的函数(即使声明为 noexcept
  • 禁止写 { *ptr } 这类依赖运行时状态的表达式——concept 检查发生在实例化前,ptr 根本没地址
  • 内置类型操作安全(如 a + bint),但自定义类型必须确保 operator+ 是 constexpr 且无副作用才稳妥

std::ranges::range 和自定义 range 的区别在哪

std::ranges::range 是标准库预定义 concept,只检查是否具备 begin()/end() 且返回迭代器类型,**不要求可遍历、不要求 begin 可 deference**:

struct fake_range {     int* begin() { return nullptr; }     int* end() { return nullptr; } }; static_assert(std::ranges::range<fake_range>); // ✅ 通过
  • 这意味着 std::ranges::range 本身不足以支撑算法(比如 std::ranges::sort 还要额外要求 random_access_iterator
  • 自己定义 range-like concept 时,应按实际需要叠加约束,例如:requires range<r> && input_iterator<iterator_t>></iterator_t></r>
  • 别盲目用 std::ranges::range 替代具体算法所需的迭代器分类——那是性能与正确性的分水岭

真正难的不是写出 concept,而是判断哪些约束该放进 concept、哪些该留在函数体内做运行时检查。过度约束会让模板失去泛化能力,约束不足又会让错误延迟到实例化深处——这没有银弹,只有对类型契约的持续校验。

text=ZqhQzanResources