当前位置: 首页 > news >正文

C++26 函数契约(Contract)概览

文章目录

    • 1. 什么是契约编程?
      • 契约编程的三大核心:
    • 2. C++26 契约编程的语法
      • 语法示例
    • 3. 契约检查模式
      • 3.1. `default` 模式
      • 3.2. `audit` 模式
      • 3.3. `axiom` 模式
      • 检查模式的设置
    • 4. 契约编程与传统 `assert` 的区别
      • 示例对比
    • 5. 契约编程的应用场景
    • 6. 注意事项
    • 7. 示例: 带契约的矩形面积计算
    • 8. 总结

契约编程(Contracts)预计成为 C++26 语言标准的一部分. 这一特性为开发者提供了一种标准化的方式来表达函数的前置条件, 后置条件以及断言, 从而显著提升代码的可读性, 可靠性和可维护性. 在本文中, 我们将深入探讨契约编程的概念, 语法, 检查模式以及实际应用场景, 帮助您快速掌握这一强大的新特性.


1. 什么是契约编程?

契约编程起源于软件工程的一种理念, 最早由 Bertrand Meyer 在 Eiffel 语言中提出. 它的核心思想是将函数的 输入约束(前置条件)输出约束(后置条件) 明确化, 形成一种程序逻辑上的契约. 这种契约不仅有助于描述函数的行为, 还能用于运行时验证, 帮助开发者快速发现问题.

契约编程的三大核心:

  1. 前置条件(Preconditions):
    函数执行前必须满足的条件, 由调用者负责. 例如, 输入参数是否有效.

  2. 后置条件(Postconditions):
    函数执行完成后必须满足的条件, 由实现者负责. 例如, 返回值是否符合预期.

  3. 断言(Assertions):
    程序执行过程中必须满足的逻辑条件, 用于验证程序内部状态是否正确.


2. C++26 契约编程的语法

C++26 提供了三个新的属性标记, 专门用于定义契约:

  • [[pre: ...]]: 定义前置条件.
  • [[post: ...]]: 定义后置条件.
  • [[assert: ...]]: 定义断言.

语法示例

以下是一个简单的除法函数, 展示了契约编程的基本用法:

#include <iostream>int divide(int a, int b)[[pre:b != 0]]                   // 前置条件: 除数不能为零[[post result:result == a / b]]  // 后置条件: 返回值必须等于 a / b
{return a / b;
}int main() {divide(10, 0);  // 触发前置条件失败, 运行时终止return 0;
}

输出:

Program returned: 139
contract violation in function divide at /app/example.cpp:3: b != 0
terminate called without an active exception
Program terminated with signal: SIGSEGV

在 Compiler Explorer 中查看


3. 契约检查模式

契约编程不仅仅是简单的运行时检查, 它通过 检查模式 提供了灵活的验证方式. C++26 提供了三种检查模式:

3.1. default 模式

  • 默认模式, 用于开发阶段.
  • 对所有契约条件进行验证, 如果检查失败, 程序终止并报告错误.
  • 适合日常开发和调试.

3.2. audit 模式

  • 加强模式, 用于深度检查.
  • 包括性能敏感路径中的契约条件.
  • 性能开销较高, 适合可靠性要求高的复杂场景.

3.3. axiom 模式

  • 假设模式, 用于生产环境.
  • 假定所有契约条件始终成立, 不进行运行时检查.
  • 编译器可以利用这些契约条件优化代码, 例如移除不必要的分支.

检查模式的设置

可以通过编译器选项控制契约检查模式, 例如:

  • GCC 示例
    g++ -std=c++26 -fcontracts -fcontract-build-level=default main.cpp   # 使用 default 模式
    g++ -std=c++26 -fcontracts -fcontract-build-level=audit main.cpp    # 使用 audit 模式
    g++ -std=c++26 -fcontracts -fcontract-build-level=off main.cpp    # 使用 axiom 模式
    

4. 契约编程与传统 assert 的区别

C++26 的契约与传统的 assert 语句在设计目标和功能上有显著差异:

特性契约(Contracts)assert
目标定义函数的行为约定, 明确调用和实现责任.临时检查某些条件是否满足, 主要用于调试.
语法专用关键字: [[pre:]], [[post:]].标准库函数: assert(condition).
范围全局行为约定, 包括前置条件, 后置条件和断言.局部检查, 仅在代码块中验证条件.
运行时模式支持 default, auditaxiom 模式.只有简单的 NDEBUG 开关控制检查启用.

示例对比

契约:

int divide(int a, int b)[[pre: b != 0]]               // 契约描述行为约定
{return a / b;
}

assert:

int divide(int a, int b) {assert(b != 0);               // 仅用于调试阶段的检查return a / b;
}

5. 契约编程的应用场景

  1. 高可靠性系统:

    • 航空航天, 自动驾驶, 金融系统等需要严格保证代码正确性.
    • 契约能够防止运行时出现未定义行为.
  2. 公共 API 开发:

    • 通过契约明确调用者的责任和函数行为, 为用户提供清晰的文档.
  3. 复杂算法实现:

    • 使用契约验证中间结果, 确保算法逻辑正确.
  4. 生产环境优化:

    • 在生产模式下切换到 axiom 模式, 移除所有运行时检查, 提升性能.

6. 注意事项

  1. 运行时开销:

    • defaultaudit 模式会增加运行时检查开销, 在性能敏感场景需谨慎使用.
  2. 编译器支持:

    • C++26 的契约编程尚未被所有主流编译器完全支持, 需要注意工具链的更新情况.
  3. 与异常处理的区别:

    • 契约不用于替代异常处理, 它更注重逻辑验证而非用户友好性.

7. 示例: 带契约的矩形面积计算

以下代码展示如何为类方法添加契约:

#include <iostream>struct Rectangle {int width;int height;int area() const[[pre:width > 0 && height > 0]]  // 前置条件: 长和宽必须为正[[post result:result > 0]]       // 后置条件: 面积必须为正{return width * height;}
};int main() {Rectangle rect{5, 10};std::cout << "Area: " << rect.area() << std::endl;Rectangle invalid{-5, 10};std::cout << invalid.area(); // 触发前置条件失败return 0;
}

在 Compiler Explorer 中查看代码

注意在三种不同模式下的输出不尽相同:

  1. default/audit:
    contract violation in function Rectangle::area at /app/example.cpp:7: width > 0 && height > 0
    terminate called without an active exception
    Program terminated with signal: SIGSEGV
    Area: 50
    
  2. axiom:
    Area: 50
    -50
    

8. 总结

契约编程通过前置条件, 后置条件和断言, 为开发者提供了一个全面的工具来提升代码质量. 在开发阶段, 契约编程有助于快速发现问题; 在生产环境中, 通过合理选择检查模式, 可以兼顾性能与安全性.

契约编程不仅是一种语言特性, 更是现代软件开发中提升代码可靠性的重要工具. 随着编译器支持的逐步完善, 它将成为每位 C++ 开发者的得力助手!

http://www.lryc.cn/news/516188.html

相关文章:

  • Flink CDC 自定义函数处理 SQLServer XML类型数据 映射 doris json字段方案
  • F.interpolate函数
  • 华为交换机---自动备份配置到指定ftp/sftp服务器
  • nginx学习之路-nginx配置https服务器
  • UCAS 24秋网络认证技术 CH10 SSL 复习
  • 【linux内核分析-存储】EXT4源码分析之“文件删除”原理【七万字超长合并版】(源码+关键细节分析)
  • 代码随想录 day62 第十一章 图论part11
  • springboot571基于协同过滤算法的私人诊所管理系统(论文+源码)_kaic
  • Uniapp Android 本地离线打包(详细流程)
  • vite+vue3动态引入资源文件(问题已解决但离了个大谱)
  • 通过 4 种方式快速将音乐从 iPod 传输到 Android
  • ArcGIS中怎么把数据提取到指定范围(裁剪、掩膜提取)
  • 【Vaadin flow 实战】第3讲-快速上手构建VaadinFlow+Springboot的全栈web项目
  • HBase Cassandra的部署和操作
  • 用户界面软件01
  • 【云原生】Docker Compose 从入门到实战使用详解
  • 【ShuQiHere】使用 SCP 进行安全文件传输
  • 海康威视H5player问题汇总大全
  • 力扣23.合并K个升序链表
  • 【C 语言指针篇】指针的灵动舞步与内存的神秘疆域:于 C 编程世界中领略指针艺术的奇幻华章
  • 游戏关卡设计的常用模式
  • 在一台服务器上使用docker运行kafka集群
  • Apache Celeborn 在B站的生产实践
  • JOIN 和 OUTER JOIN,SQL中常见的连接方式
  • Vue2: table加载树形数据的踩坑记录
  • 电子信息硕士面试经验
  • dns网址和ip是一一对应的吗?
  • springboot3 redis 常用操作工具类
  • Java工程师实现视频文件上传minio文件系统存储及网页实现分批加载视频播放
  • Redis(二)value 的五种常见数据类型简述