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

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.kler.cn/a/468718.html

相关文章:

  • Java最新面试题(全网最全、最细、附答案)
  • 初学STM32 --- USMART
  • js复制数据到剪切板
  • 数据挖掘教学指南:从基础到应用
  • Spring AMQP ----注解篇
  • 测试用例颗粒度说明
  • 计算机网络 —— 网络编程(TCP)
  • 基于Web的足球青训俱乐部管理后台系统的设计与开发源码(springboot+mysql+vue)
  • JAVA开发中 MyBatis XML 映射文件 的作用
  • LabVIEW语言学习过程是什么?
  • 轻量级通信协议 JSON-RPC 2.0 详解
  • GraalVM:云原生时代的Java虚拟机
  • QPainter
  • 从零开始学TiDB(8) TiFlash 主要架构
  • 通过串口通信控制led灯的亮灭
  • 如何在centos中进行有效的网络管理
  • 基于ESP32的桌面小屏幕实战[5]:PCB下单
  • 深入Android架构(从线程到AIDL)_12 Android UI 单线程程序
  • AIGC生图实战技巧分享
  • iptable限制多个端口出站
  • C++直接内存管理new和delete
  • Kubernetes——part4-1 Kubernetes集群 服务暴露 Nginx Ingress Controller
  • frameworks 之 WallpaperManagerService 壁纸管理
  • spring mvc源码学习笔记之三
  • c语言的文件操作与文件缓冲区
  • 抖音生活服务2024年覆盖线下商家门店超610万,交易额同比提升81%