C++之assert推荐用法及注意事项

 更新时间:2024年07月10日 09:16:08   作者:流星雨爱编程  
这篇文章主要给大家介绍了关于C++之assert推荐用法及注意事项的相关资料,assert 是一个预处理宏,用于在运行时检查表达式是否为真,文中通过代码介绍的非常详细,需要的朋友可以参考下

1.引言

assert在 <assert.h> 头文件中定义,其作用是如果它的提交呢返回错误,则终止执行。

原型定义:

    #include <assert.h>
    void assert(int expression);

assert的作用是先计算表达式expression,如果其值为假(即为0),那么它向stderr打印一条错误信息,然后通过调佣abort来终止程序运行。

通常,它用于检查在程序执行过程中应始终为真的条件。即当违反真理时, 触发断言. 比如1+1=2,当1+1≠2时触发断言。比如,

    assert(false); // 中断程序;
    assert(0 && "the code is error!"); // 中断并打印信息

2.简单示例

#include <assert.h> // header

int divide(int numerator, int denominator) {
    assert(denominator != 0); // 确保分母不为零
    return numerator / denominator;
}

int main() {
    int result = divide(10, 2);
    // ...
    return 0;
}

在上面的示例中,assert 宏确保分母永远不为零。如果分母为零,程序将以指示文件名、行号和失败的表达式的错误消息终止。

3.推荐使用方法

经常在C++程序中,以便于在开发过程中进行断言检查,而在发布版本中则禁用这些检查。

推荐写成如下方式,编写一个assert_utils.h文件,内容如下:

#pragma once

#ifndef RELEASE
#include <cassert>
#define ASSERT(f) assert(f) 
#else
#define ASSERT(f) ((void)0)
#endif

下面是对这段代码的解释:

1) #pragma once:这是一个预处理指令,用于确保头文件只被包含一次。它的效果取决于编译器,但大多数现代编译器都支持它。

2) #ifndef RELEASE:这是一个预处理指令,用于检查是否定义了RELEASE宏。如果没有定义,编译器将继续执行下面的代码块。

3) #include <cassert>:如果RELEASE宏没有定义,编译器将包含cassert头文件,这是标准C++库的一部分,提供了assert宏。

4) #define ASSERT(f) assert(f):定义了一个宏ASSERT,它接受一个表达式f作为参数。当ASSERT被调用时,它将执行assert(f),如果f为假(即表达式结果为0),程序将终止运行,并显示一个断言失败的消息。

5) #else:如果RELEASE宏被定义,编译器将跳过上面的代码块,执行下面的代码。

6) #define ASSERT(f) ((void)0):在发布版本中,ASSERT宏被定义为一个空操作,即什么都不做。这样,在发布版本的程序中,所有ASSERT调用都不会执行任何操作,从而避免了性能损耗和潜在的程序中断。

7) #endif:结束#ifndef RELEASE条件编译块。

4.常见使用场景和注意事项

4.1.检查参数的合法性

示例如下:

int  sendData(const char* pData, int len)
{
    //[1] 校验参数的合法性
    assert(pData);
    assert(len > 0)

    //[2]
    //...
}

每个assert只检验一个条件,因为同时检验多个条件时,如果断言失败,无法直观的判断是哪个条件失败,如:

assert(pData && len > 0);  //这样不好

4.2.assert不能使用改变环境的语句

因为assert只在DEBUG环境中生效,如果这么做,会导致DEBUG和RELEASE的逻辑不一致。如:

assert(i++ < 100);

在DEUG环境下,每次都会执行i++,而在RELEASE环境中不会执行i++,这样就会导致两种环境下的执行结果不一样,正确的写法是:

assert(i < 100);
i++;

4.3.在未知的逻辑中添加assert(false)

如:

//根据业务类型处理业务
void  dealPro(int type)
{
    if (type == 0){
        // 文本业务
    }else if (type == 1){
        // 地图业务
    }else if (type == 2){
        // 测试业务
    }else{
        assert(false);
    }
}

上述代码,我们的业务逻辑只涉及到文本、地图、测试业务,其它的都是非法业务,添加assert(false),一旦type不等与1、2、3,就会出现断言错误,查看程序堆栈,这样就非常快速的定位程序是哪个地方出现逻辑错误了,这个也是一个非常好的调试技巧。

5.总结

总之,assert是一种在开发过程中快速检测程序错误的有效工具,但在发布的产品代码中通常被禁用以避免性能影响。开发者可以根据需要使用assert或其他错误处理机制来确保程序的正确性和健壯性。

到此这篇关于C++之assert推荐用法及注意事项的文章就介绍到这了,更多相关C++ assert用法内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C读txt到二维数组的实现方法

    C读txt到二维数组的实现方法

    下面小编就为大家带来一篇C读txt到二维数组的实现方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-12-12
  • 4组C语言中顺序读写文件的函数分享

    4组C语言中顺序读写文件的函数分享

    这篇文章主要为大家详细介绍了4组C语言中实现顺序读写文件的函数,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2023-03-03
  • Qt使用QChart实现动态显示温度变化曲线

    Qt使用QChart实现动态显示温度变化曲线

    Qt的QChart是一个用于绘制图表和可视化数据的类,提供了一个灵活的、可扩展的、跨平台的图表绘制解决方案,所以本文就将使用QChart实现动态显示3个设备的温度变化曲线,感兴趣的可以了解一下
    2023-06-06
  • 浅谈VC中预编译的头文件放那里的问题分析

    浅谈VC中预编译的头文件放那里的问题分析

    本篇文章是对VC中预编译的头文件放那里的问题进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • C++ STL vector的模拟实现

    C++ STL vector的模拟实现

    这篇文章主要介绍了C++ STL vector的模拟实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-05-05
  • C语言数据结构之单向链表详解

    C语言数据结构之单向链表详解

    单向链表(单链表)是链表的一种,其特点是链表的链接方向是单向的,对链表的访问要通过顺序读取从头部开始。本文将为大家详细讲讲单向链表的实现与使用,需要的可以参考一下
    2022-08-08
  • C语言趣味编程之平分七筐鱼

    C语言趣味编程之平分七筐鱼

    这篇文章介绍了C语言趣味编程之平分七筐鱼,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-11-11
  • 使用VC6.0对C语言程序进行调试的基本手段分享

    使用VC6.0对C语言程序进行调试的基本手段分享

    这篇文章主要介绍了用VC6.0开发c语言程序的时候调试代码的一些小技巧,需要的朋友可以参考下
    2013-07-07
  • C++实现LeetCode(115.不同的子序列)

    C++实现LeetCode(115.不同的子序列)

    这篇文章主要介绍了C++实现LeetCode(115.不同的子序列),本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • C语言编程入门必背的示例代码整理大全

    C语言编程入门必背的示例代码整理大全

    这篇文章主要为大家整理并介绍了C语言编程必背的示例代码大全,有需要的朋友可以借鉴参考下,希望能够有所帮助
    2021-11-11

最新评论