首页 > 编程语言 >C++26 函数契约(Contract)概览

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

时间:2025-01-05 19:33:03浏览次数:3  
标签:26 检查 int 编程 C++ Contract 契约 模式

文章目录

契约编程(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++ 开发者的得力助手!

标签:26,检查,int,编程,C++,Contract,契约,模式
From: https://blog.csdn.net/arong_xu/article/details/144949179

相关文章

  • C++ 实现定时器
    冬天的午后,寒意略显温柔,不像晨时那样刺骨,也不像入夜之时的冰冷。阳光倾斜落在阳台上。想必它是耀眼的,照在屋外树梢上仅剩的几片叶子上,闪闪发光,有些晃眼。学习自:零声教育的视频1.什么是定时器定时器是一种用于在未来某个时间点执行某个任务的机制。在操作系统中,定时器是一......
  • 【C++动态规划】2088. 统计农场中肥沃金字塔的数目|2104
    本文涉及知识点C++动态规划LeetCode2088.统计农场中肥沃金字塔的数目有一个矩形网格状的农场,划分为m行n列的单元格。每个格子要么是肥沃的(用1表示),要么是贫瘠的(用0表示)。网格图以外的所有与格子都视为贫瘠的。农场中的金字塔区域定义如下:区域内格子数......
  • SM260331btnQD
    usingSystem;usingSalien.Utility.SUWF;usingSalien.Utility;usingClsPub;namespaceSM260331btnQD{publicclassSM260331btnQD:ISuwfBus{privateSlnSuwfPage_page;publicvoidInitial(SlnSuwfPagepage){......
  • 4.3 C++对象模型和this指针
    4.3.1成员变量和成员函数分开存储只有非静态成员变量才属于类的对象上1.空对象占用的内存空间为1,为了区分空对象占用的位置2.非静态成员变量占用4个内存空间,属于类的对象上的3.静态成员变量static不占对象空间,不属于类的对象上的4.函数不占对象空间,所有函数共享一个......
  • 《 C++ 点滴漫谈: 十七 》编译器优化与 C++ volatile:看似简单却不容小觑
    摘要本文深入探讨了C++中的volatile关键字,全面解析其基本概念、典型用途以及在现代编程中的实际意义。通过剖析volatile的核心功能,我们了解了它如何避免编译器优化对硬件交互和多线程环境中变量访问的干扰。同时,文章分析了volatile的局限性,如缺乏线程安全保障,并介......
  • C++ 前缀和
    有一个数组{2,1,3,6,4},询问三次结果:a[5]={2,1,3,6,4}1.数组第1到第2个元素的和是多少?2.数组第1到第3个元素的和是多少?3.数组第2到第4个元素的和是多少?原始方法(无前缀和):1#include<iostream>2#include<stdio.h>3usingnamespacestd;4intmain(){5......
  • C++前缀和
    有一个数组{2,1,3,6,4},询问三次结果:a[5]={2,1,3,6,4}1.数组第1到第2个元素的和是多少?2.数组第1到第3个元素的和是多少?3.数组第2到第4个元素的和是多少?  没有用前缀和的原始用法:1#include<iostream>2#include<stdio.h>3usingnamespacestd;4intma......
  • 计算机毕设项目28r41260+springboot基于Hadoop的豆瓣电子图书推荐系统统,计算机毕业生
    基于Hadoop实现的豆瓣电子图书推荐系统的设计与实现摘 要随着开数字化阅读的普及,豆瓣电子图书推荐系统应运而生,旨在为用户提供个性化的阅读体验。基于Hadoop的强大数据处理能力,该系统能够有效处理海量用户数据和书籍信息,通过复杂的算法模型为用户推荐高质量的内容。管理员......
  • C++版AI猜数
    源码#include<iostream>#include<ctime>usingnamespacestd;inta[17]={0,1,3,5,7,9,11,13,15,17,19,21,23,25,27,29,31};intb[17]={0,2,3,6,7,10,11,14,15,18,19,22,23,26,27,30,31};intc[17]={0,4,5,6,7,12......
  • C++中的 多维数组、锯齿数组
    多维数组定义:多维数组可以看作是数组的数组,通过在定义时指定每个维度的大小来创建。下面以三维数组为例。访问:使用多个索引来访问数组中的元素,索引从0开始。销毁:对于栈上定义的多维数组,当作用域结束时会自动销毁;对于堆上动态分配的多维数组,需要手动释放内存。#include<iost......