C++20中的Concepts(概念)有什么用?(对模板参数增加类型约束)

Concepts 能提前发现模板错误,如 std::sort 对迭代器类型约束不满足时立即报错而非展开大量晦涩错误;自定义 Concept 应描述操作可行性而非贴标签,用 requires 块检查表达式合法性与返回类型;约束位置影响语义:模板参数级最清晰,函数尾部用于 SFINAE,if constexpr 用于编译期分支;Concepts 不提升运行时性能,但增强泛型安全性与接口契约明确性。

Concepts 能帮你提前发现模板错误

没有 Concepts 时,std::sort 接收一个随机访问迭代器,但如果你传入 std::list::iterator(只支持双向遍历),编译器不会在第一眼报错,而是深入到内部实现里展开大量模板实例化后才爆出几十行晦涩的错误信息。用 Concepts 后,约束写在函数声明处:

template
void my_sort(Iter first, Iter last);
——传错类型立刻提示“std::list::iterator does not satisfy std::random_access_iterator”,错误位置精准、信息直白。

自定义 Concept 要写清楚“它必须能做什么”

Concept 不是给类型贴标签,而是描述一组操作是否可用。比如想约束某个类型支持加法和取负:

template
concept AddableAndNegatable = 
    requires(T a, T b) {
        { a + b } -> std::same_as;
        { -a } -> std::same_as;
    };
注意三点:

  • requires 块列出表达式,不是检查成员名
  • { a + b } 表示“这个表达式合法”,-> std::same_as 才检查返回类型
  • 不要写 requires std::is_arithmetic_v 这类静态断言——Concept 的意义在于约束接口行为,不是硬塞类型列表

Concept 约束位置不同,语义完全不同

同一个 Concept 放在不同地方,效果差很远:

  • 作为模板参数约束:template → 编译期筛选,不匹配直接不参与重载解析
  • 放在函数声明末尾:void f(T t) requires AddableAndNegatable; → 属于 SFINAE 友好约束,失败时该重载被静默丢弃
  • 用在 if constexpr 里:if constexpr (AddableAndNegatable) → 运行时不可见的编译期分支,适合写 fallback 逻辑
混用这三种写法容易导致重载决议意外失败,尤其当多个约束条件交叉时,建议优先用模板参数级约束,语义最清晰。

别指望 Concepts 提高性能,但它让泛型更安全

Concepts 本身不改变生成代码,也不影响运行时开销

。它的价值全在编译期:

  • 避免无意中实例化不支持的操作(比如对 std::vector 调用 data()
  • 让库作者能明确写出“这个算法要求什么”,而不是靠文档或用户试错
  • 配合 std::ranges 等新设施,才能真正启用基于范围的、可组合的算法
最容易被忽略的一点:Concept 检查的是“当前上下文可见的接口”。如果某个类型在头文件里只前向声明了,但关键操作定义在 cpp 文件里,Concept 判断会失败——这时候不是约束太严,而是 ODR 或包含顺序出了问题。