异常处理/CC++ 中 assert 断言 应用实践和注意事项

文章目录

  • 概述
  • assert 本质浅析
  • Release版本下的assert是否生效
    • 默认设置下 QtCreator环境 assert 过程
    • 默认配置下 VS环境 assert 过程
    • 配置VS发布模式下的断言生效
    • VS环境Release版本的UI程序
    • Release下请当我不生效
  • 请勿滥用assert
    • 导致逻辑错误
    • 再强调'不要在assert内执行逻辑功能'
    • 怎敢默认release下绝不会发生此错误?
    • 要不要在Release版本下使用断言
    • 使用assert的其他建议
  • 静态断言
  • 自定义断言
  • 小结

概述

本文主要讲解了 assert 断言机制,在编程中的作用和注意事项,如 assert 的工作原理、Release程序版本下的断言生效问题、为什么要杜绝在assert内执行逻辑、如何自定义断言等。断言机制是在开发和调试阶段快速发现程序中的错误和逻辑问题的重要手段,它可以帮助开发人员在程序中插入检查点,以验证程序的正确性和健壮性,一旦发现断言失败,开发人员可以通过查看错误消息和堆栈跟踪来定位和解决问题。

转载请标明原文链接,
https://blog.csdn.net/quguanxin/category_6223029.html

assert 本质浅析

标准C/C++库中的 assert 并不是一个函数,事实上,它是个宏,其用法像是一种"契约式编程",其表达的意思是,程序在我的假设条件下,能够正常良好的运作,否则就告警并调用 abort 终止程序。其使用场景大概是这样的,大多数情况下,我们要进行验证的假设,只是属于偶然性事件,又或者我们仅仅想测试一下,一些最坏情况是否发生。

int main() {printf("当前源文件名:%s, 当前代码行号:%s\n", FILE_NAME, __LINE__);assert(0 == 1);system("pause"); return 0;
}

在这里插入图片描述
通过上述测试,可以猜测,assert 内部极有可能是封装了 FILELINE 预处理宏的。看一下源码,

#ifdef NDEBUG#define assert(expression) ((void)0)
#else_ACRTIMP void __cdecl _wassert(_In_z_ wchar_t const* _Message,_In_z_ wchar_t const* _File,_In_   unsigned       _Line);#define assert(expression) (void)(                                                       \(!!(expression)) ||                                                              \(_wassert(_CRT_WIDE(#expression), _CRT_WIDE(__FILE__), (unsigned)(__LINE__)), 0) \)
#endif

与我们《》自定义函数的使用方法一致,先以 char* 和 int 为参数类型,定义函数,然后,使用宏封装此函数,这样,LINEFILE 就可以用来代表宏函数(assert) 的调用位置。

Release版本下的assert是否生效

验证下,assert断言在发布模式下是否起作用-/-
能有多灵验呢,能否在崩溃的时候,显示出来—
实际运行中,我在发布版里头没怎么体现出来呢?
最好的办法还是为自己的软件编写错误处理框架–

默认设置下 QtCreator环境 assert 过程

void Widget::on_pushButton_clicked()
{ui->label->setText("before assert");//assert(false);  //Line:22Q_ASSERT(false);  //Line:23ui->label->setText("after assert");
}

使用C库的assert测试
编译Debug版本和Release版本,脱离开发环境进行执行测试,结果是都可以准确的定位到断言所在的代码行。因此我们初步得出的结论是,在QtCreator开发环境下,默认的情况下,Release版本的程序中,C库中的assert是可以生效的。

使用Qt库的Q_ASSERT测试
使用Qt自带的Q_ASSERT断言宏时,经过测试,在Release版本的程序中并不生效。

#if !defined(Q_ASSERT)
#  if defined(QT_NO_DEBUG) && !defined(QT_FORCE_ASSERTS)
#    define Q_ASSERT(cond) static_cast<void>(false && (cond))
#  else
#    define Q_ASSERT(cond) ((cond) ? static_cast<void>(0) : qt_assert(#cond, __FILE__, __LINE__))
#  endif
#endif/*The Q_ASSERT macro calls this function when the test fails.
*/
void qt_assert(const char *assertion, const char *file, int line) Q_DECL_NOTHROW
{QMessageLogger(file, line, nullptr).fatal("ASSERT: \"%s\" in file %s, line %d", assertion, file, line);
}

通过上述Qt框架下的宏定义,我们可以看出,如果没有定义QT_FORCE_ASSERTS强制使用断言,而且定义了QT_NO_DEBUG(估计它在Release编译模式下会有定义),Q_ASSERT相当于没有定义。

默认配置下 VS环境 assert 过程

纯 C++ 代码

#include <iostream>
#include <assert.h>int main() {int i = 0;std::cout << "Hello World!\n";assert(i != 0);std::cout << "Hello weifang!\n";system("pause");
}

在Debug模式下运行,断言如下。在Release模式下,断言不生效。
在这里插入图片描述

配置VS发布模式下的断言生效

还是上一小节中的项目,我们查看其项目属性,C/C++,预处理器,预处理器定义-
Debug配置:
在这里插入图片描述
Release配置:
在这里插入图片描述
我们针对Release模式下的预处理器定义,删除其中的NDEBUG宏,重新运行Release版本,
在这里插入图片描述
如上,只要删除NDEBUG宏定义,则assert在Release模式下也是生效的

注意一个现象,使用NDEBUG宏和删除该宏的情况下,编译生成的可执行文件的大小和占用空间大小保持一致不变。即使是在Release模式下预编译器配置上增加_DEBUG宏,其效果也是与不设置NDEBUG宏一致,与Debug模式下配置_DEBUG宏是不一样的。

另外,在 C++ 中,assert 宏通常在调试模式下才会生效,而在发布模式下会被编译器忽略掉。如果你希望在发布模式下也启用 assert 断言,可以在 “预处理器定义” 字段中,删除NDEBUG宏,或进一步添加 _DEBUG宏定义。如此便确保 _DEBUG 宏在发布模式和调试模式下都被定义。要注意的是, assert 断言主要用于在开发和调试阶段发现问题。在发布版本中,最好使用其他方式进行错误检查和处理,如异常处理、返回错误码或输出错误日志等。

VS环境Release版本的UI程序

在脱离开发环境的情况下,运行上一节的控制台测试程序。Debug版本程序(配有_DEBUG)的运行会弹出提示。Release版本程序(删除NDEBUG宏)的运行,在控制台有断言提示,没有弹窗提示,且控制台在数秒后自动退出。本小节我们进一步看看,在没有NDEBUG的情况下,R版的UI程序会是什么执行现象。

在VS下新建Qt项目,D版和R版的默认预处理器定义一致为$(Qt_DEFINES_);%(PreprocessorDefinitions),这里我暂时没有在Qt_DEFINES_变量中看到_DEBUG或NDEBUG宏的身影。它一定藏在其他地方了,我们暂不深究。

TestAssertUI::TestAssertUI(QWidget *parent) : QWidget(parent) , ui(new Ui::TestAssertUIClass())
{ui->setupUi(this);//connect(ui->pushButton, &QPushButton::clicked, []{int i = 0;assert(i != 0);qDebug() << "The assertion is not in effect";});
}

编译并执行my_D版本的程序,
在这里插入图片描述
编译并执行my_R版本的程序,断言并未生效。那,我们在my_R预处理器定义中增加 _DEBUG宏定义,重新编译执行。结果,断言依然是被优化不执行的。

还好我眼尖,在预处理器定义项目的下面发现了"取消预处理器定义"这个配置项,我将NDEBUG配置进去。
在这里插入图片描述
重新编译的过程提示,cl : 命令行 warning D9025: 正在重写“/DNDEBUG”(用“/UNDEBUG”),这一看就是起作用的样子啊。编译后执行,果然,与上述DEBUG版程序的弹窗告警一致

重点总结,
在使用VS做集成开发环境时,如果想使得R版本程序中C库的assert生效,需要在项目属性配置,C/C++,预处理器,取消预处理器,这个配置项中配置取消 “NDEBUG宏”。(仅)在预处理定义中增加_DEBUG宏定义是无效的,这可能是因为NDEBUG宏等项目属性中隐含的定义,会在编译过程靠后的阶段进行加载和覆盖。
在VS开发环境下,针对Release配置。其中非Qt项目,其预处理器定义中会直接标明NDEBUG,若想打开assert调试功能,只需要删除即可。针对Qt项目,由于NDEBUG没有直接定义在预处理器配置项中,你需要在取消预处理定义这个配置项中添加NDEBUG宏。

Release下请当我不生效

为了保险起见,一种可行的做法是,认为任何IDE下的任何ASSERT语句,是不生效的,不执行的。

请勿滥用assert

虽然 assert 方便好用,但勿滥用!

导致逻辑错误

知道assert在release模式下"可能"不被执行,可还是手欠。为了图省事,伪代码如下:

//!!危险!!
assert(SomethingBegin());
//函数定义
bool SomethingBegin()
{if (s_bBuseFlage)return false;//重置上下文	s_count = 0; s_bBuseFlage = false;return true;
}

如上,我assert了一个函数的执行返回值。在VS环境下,当编译release版本时,上述assert语句将被优化掉,也就是说其中的含有逻辑功能的函数将不会被执行,这必然导致运行异常。Debug下没有问题,release时抓瞎,通常会火烧眉毛。

再强调’不要在assert内执行逻辑功能’

//这条代码在R模式下可能不执行,从而导致逻辑异常assert(m_pObserver->Subscribe_Open(_ID_STREAM_INFO));

上述是曾经在实际项目中犯下的错误,而且那还是在总结过assert使用注意事项,已经很清楚assert内不可以执行逻辑代码。总归有那么几个脑子变浆糊的时刻,手欠。因此,遇到D模式和R模式执行不一致的情况,assert使用,算是一个检查点。

 //不要偷懒,只在assert中判断结果值
bOk &= m_pObserver->Subscribe_Open(_ID_STREAM_INFO);
assert(bOk);

怎敢默认release下绝不会发生此错误?

一种情形是这样的,Debug下并没有触发assert,但release下却发生了要捕获的异常,由于编码和编译原因,若release版本中assert被优化掉,那么,你相当于是放弃了对此异常情形的处理!那么就危险了,这种危险远不止是你的程序异常退出了一次,而是你没有什么可用信息去定位异常位置。

要不要在Release版本下使用断言

首先表明立场,不建议在Release版本中使用assert断言生效。你最好使用其他形式的错误处理和日志记录来补充 assert 断言,以提供更好的用户体验和容错能力。

好处:
错误检测:assert 断言是一种在运行时检测程序中的错误和异常的方法。在 Release 版本中启用 assert 断言可以帮助及早发现潜在的问题和错误,提高代码质量和可靠性。
调试信息:assert 断言通常会输出有关断言失败的相关信息,例如断言所在的文件和行号等。在 Release 版本中启用 assert 断言可以提供有用的调试信息,以便更好地理解和排查问题。
安全验证:通过启用 assert 断言,您可以在 Release 版本中对关键的安全验证进行检查。这可以帮助捕获潜在的安全漏洞和错误用法,提高程序的安全性。

坏处:
性能影响:assert 断言通常在断言失败时会导致程序终止。这会对程序的性能产生一定的负面影响。因此,在 Release 版本中启用 assert 断言可能会导致性能下降,尤其是在大规模或性能敏感的应用程序中。
用户体验:断言失败可能导致程序异常终止,这可能会对用户体验产生负面影响。在某些情况下,这可能会导致数据丢失或不可预测的行为。因此,在启用 assert 断言时需要谨慎权衡用户体验和错误检测的需求。
可移植性问题:某些平台或环境可能不支持 assert 断言,或者对其行为进行了修改。因此,在跨平台或跨环境的应用程序中,依赖于 assert 断言的特定行为可能会导致可移植性问题。

使用assert的其他建议

assert 在那些一次性执行的代码语句上可以使用,这些语句往往是非常的硬,如果一旦有错误错误,程序将没有继续运行的必要。
在那些动态频繁执行的函数中,如果使用assert来进行某些校验,往往会给自己带来不少麻烦。

虽然启用断言可能会带来性能影响和一些不太友好的用户体验,但在开发和调试阶段,启用断言可以帮助发现和修复潜在的错误和异常,从而提高代码质量和可靠性。在发布和部署阶段,需要谨慎权衡性能与错误检测需求之间的平衡。

像是动态内存申请之类的操作,强烈建议不要用assert检验返回结果。但有时候你实在懒得一坨,加个assert并打开release的DEBUG开关,也是种手段,这至少比你置之不理要好很多。因为空指针通常会导致程序毫无征兆的死掉,让你束手无策,那不仅不优雅,还会让运维和开发工程师很头疼。故,当你的软件没有完备的异常处理或日志记录机制,那就用assert做最后的救命稻草吧!

静态断言

C/C++ 中的静态断言机制(Static Assertion)是一种在编译时进行静态检查的机制,用于在编译器发现错误之前捕获问题。static_assert 接受一个编译时求值为布尔值的表达式作为参数,并在表达式为假时触发编译错误。如果表达式为真,则静态断言不会产生任何代码或运行时开销。如下,是ROS2.0 异常处理模块中的源码片段,

/// Struct wrapping a fixed-size c string used for returning the formatted error string.
typedef struct rcutils_error_string_s {/// The fixed-size C string used for returning the formatted error string.char str[RCUTILS_ERROR_MESSAGE_MAX_LENGTH];
} rcutils_error_string_t;  //该结构的长度与rcutils_error_state_t长度相同/// Struct which encapsulates the error state set by RCUTILS_SET_ERROR_MSG().
typedef struct rcutils_error_state_s {/// User message storage, limited to RCUTILS_ERROR_STATE_MESSAGE_MAX_LENGTH characters.char message[RCUTILS_ERROR_STATE_MESSAGE_MAX_LENGTH];/// __FILE__宏代表的代码文件名 File name, limited to what's left from RCUTILS_ERROR_STATE_MAX_SIZE characters after subtracting storage for others.char file[RCUTILS_ERROR_STATE_FILE_MAX_LENGTH];/// __LINE__宏代表的代码行号 Line number of error.uint64_t line_number;
} rcutils_error_state_t;  //该结构的长度与rcutils_error_string_t长度相同// make sure our math is right... //编译时进行静态断言from C++ 11
#if __STDC_VERSION__ >= 201112L
static_assert(sizeof(rcutils_error_string_t) == (  /* 1024 == 768 + 229 + 20 + 6 + 1(null terminating character) */RCUTILS_ERROR_STATE_MESSAGE_MAX_LENGTH + RCUTILS_ERROR_STATE_FILE_MAX_LENGTH + RCUTILS_ERROR_STATE_LINE_NUMBER_STR_MAX_LENGTH + RCUTILS_ERROR_FORMATTING_CHARACTERS + 1),"Maximum length calculations incorrect");
#endif

自定义断言

如在heap4中有类似如下定义,

//定义错误信息输出函数
#define vAssertCalled(charFile, intLine) AflDebugError("Error:%s,%d\r\n",charFile, intLine)
//利用 __FILE__,__LINE__ 预定义宏
#define configASSERT(x) if((x)==0) vAssertCalled(__FILE__,__LINE__)

早期在打印调试单元中定义的断言,不去中断程序执行,

//assert 自定义断言
#define AFLPRINTF_ASSERT(bAssert)  \if (!(bAssert))                \{ \printf("AFLPRINTF_ASSERT %s,%d:", FUNCNAME, __LINE__); \printf("\r\n");   \fflush(stdout);   \}  \

其他的定义可参考 《异常处理/LINEFILE 宏在调试和异常处理中的高级使用》 等文章。

小结

在百科有提到 assert.h 常用于防御式编程。,防御式编程的主要思想是:子程序应该不因传入错误数据而被破坏,哪怕是由其他子程序产生的错误数据。这种思想是将可能出现的错误造成的影响控制在有限的范围内。这里也临时总结几个使用断言的几个原则,
(1)使用断言捕捉不应该发生的非法情况。不要混淆非法情况与错误情况之间的区别,后者是必然存在的并且是一定要作出处理的。
(2)使用断言对函数的参数进行确认。
(3)在编写函数时,要进行反复的考查,并且自问:"我打算做哪些假定?"一旦确定了的假定,就要使用断言对假定进行检查。
(4)一般教科书都鼓励程序员们进行防错性的程序设计,但要记住这种编程风格会隐瞒错误。当进行防错性编程时,如果"不可能发生"的事情的确发生了,则要使用断言进行报警。
ASSERT 是一个调试程序时经常使用的宏,在程序运行时它计算括号内的表达式,如果表达式为FALSE (0), 程序将报告错误,并终止执行。如果表达式不为0,则继续执行后面的语句。这个宏通常原来判断程序中是否出现了明显非法的数据,如果出现了终止程序以免导致严重后果,同时也便于查找错误。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/10929.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【UnityRPG游戏制作】Unity_RPG项目_PureMVC框架应用

&#x1f468;‍&#x1f4bb;个人主页&#xff1a;元宇宙-秩沅 &#x1f468;‍&#x1f4bb; hallo 欢迎 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! &#x1f468;‍&#x1f4bb; 本文由 秩沅 原创 &#x1f468;‍&#x1f4bb; 收录于专栏&#xff1a;就业…

Vue-watch监听器

监听器 watch侦听器&#xff08;监视器&#xff09;简单写法完整写法 watch侦听器&#xff08;监视器&#xff09; 作用&#xff1a;监视数据变化&#xff0c;执行一些业务逻辑或异步操作 语法&#xff1a; watch同样声明在跟data同级的配置项中简单写法&#xff1a; 简单类型…

C++ 中的 lambda 表达式

1.概念 lambda表达式实际上是一个匿名类的成员函数&#xff0c;该类由编译器为lambda创建&#xff0c;该函数被隐式地定义为内联。因此&#xff0c;调用lambda表达式相当于直接调用匿名类的operator()函数&#xff0c;这个函数可以被编译器内联优化&#xff08;建议&#xff0…

地图涟漪效果

参考API echarts图表集 useEcharts.js import { onBeforeUnmount, onDeactivated } from "vue"; // import * as echarts from "echarts";/*** description 使用 Echarts (只是为了添加图表响应式)* param {Element} myChart Echarts实例 (必传)* param …

AcWing-168生日蛋糕-搜索/剪枝

题目 思路 表面积和体积公式&#xff1a;以下分析参考自&#xff1a;AcWing 168. 生日蛋糕【图解推导】 - AcWing&#xff1b;AcWing 168. 关于四个剪枝的最清楚解释和再次优化 - AcWing 代码 #include<iostream> #include<cmath> using namespace std;const in…

【爬虫基础1.1课】——requests模块上

目录索引 requests模块的作用&#xff1a;实例引入&#xff1a; 特殊情况&#xff1a;锦囊1&#xff1a;锦囊2: 这一个栏目&#xff0c;我会给出我从零开始学习爬虫的全过程。感兴趣的小伙伴可以关注一波&#xff0c;用于复习和新学都是不错的选择。 那么废话不多说&#xff0c…

C语言学习(九)多文件编程 存储类型 结构体

目录 一、多文件编程&#xff08;一&#xff09;不写头文件的方方式进行多文件编程 &#xff08;二&#xff09;通过头文件方式进行多文件编程&#xff08;1&#xff09;方法&#xff08;2&#xff09;头文件守卫 &#xff08;三&#xff09; 使用多文件编程实现 - * / 功能 二…

HC-06 蓝牙串口从机 AT 命令详解

HC-06 蓝牙串口从机 AT 命令详解 要使用 AT 命令&#xff0c;首先要知道 HC-06 的波特率&#xff0c;然后要进入 AT 命令模式。 使用串口一定要知道三要素&#xff0c;一是波特率&#xff0c;二是串口号&#xff0c;三是数据格式, HC-06只支持一种数据格式: 数据位8 位&#…

HTTP 连接详解

概述 世界上几乎所有的 HTTP 通信都是由 TCP/IP 承载的&#xff0c;客户端可以打开一条TCP/IP连接&#xff0c;连接到任何地方的服务器。一旦连接建立&#xff0c;客户端和服务器之间交换的报文就永远不会丢失、受损或失序 TCP&#xff08;Transmission Control Protocol&…

97. 交错字符串-----回溯、动态规划

题目链接 97. 交错字符串 - 力扣&#xff08;LeetCode&#xff09; 解答 递归回溯 题目所述为两个字符串交替组成第三个字符串&#xff0c;之前好像做过相似的题目&#xff0c;直接联想到可以考虑使用递归回溯的做法&#xff0c;让字符串s1和字符串s2分别作为起始字符串&…

Mybatis-Plus大批量插入数据到MySQL

MyBatis-Plus的saveBatch方法 GetMapping("/save1") public void save1() {// 数据准备List<MallOrder> orderList getMallOrderList();// mybatis-pluslong start System.currentTimeMillis();mallOrderService.saveBatch(orderList);System.out.println(&…

计算机服务器中了360后缀勒索病毒怎么解密,360后缀勒索病毒恢复

计算机网络技术的不断发展与应用&#xff0c;为企业的生产运营提供了极大便利&#xff0c;大大提高了企业的办公效率&#xff0c;为企业的生产运营注入了新的动力&#xff0c;但网络是一把双刃剑&#xff0c;在为企业提供便利的同时&#xff0c;也为企业的数据安全带来严重威胁…

google test 使用指南

目录 测试项目 calculator.h calculator.cpp test01.cpp 创建新项目 选择Google Test 选择要测试的项目 pch.cpp 加入依赖 设为启动项目 ​编辑 运行 ​编辑 关键点 测试项目 calculator.h #ifndef __CALCULATOR_H__ #define __CALCULATOR_H__#include <i…

Linux操作系统中管理磁盘的另外一种操作方式。即LVM——逻辑卷管理操作

在Linux操作系统中管理磁盘的一种方法名称——LVM&#xff0c;这种管理磁盘的优势。 1.使用LVM去管理磁盘可以在不影响原来数据的前提下去扩容磁盘空间或者是缩减磁盘空间。 在LVM中除了上层逻辑券可以扩容&#xff0c;下层的券组也可以扩容。 2.使用LVM管理的磁盘支持快照功…

MySQL中的子查询

子查询,在一个查询语句中又出现了查询语句 子查询可以出现在from和where后面 from 表子查询(结果一般为多行多列)把查询结果继续当一张表对待 where 标量子查询(结果集只有一行一列)查询身高最高的学生,查询到一个最高身高 列子查询(结果集只有一行多列) 对上表进行如下操作 …

韩顺平0基础学Java——第10天

p202-233 类与对象&#xff08;第七章&#xff09; 成员方法 person类中的speak方法&#xff1a; 1.public表示方法是公开的 2.void表示方法没有返回值 3.speak&#xff08;&#xff09;中&#xff0c;speak表示方法名&#xff0c;括号是形参列表。 4.大括号为方法体&am…

WPF之多种视图切换

1&#xff0c;View切换&#xff0c;效果呈现 视图1 视图2 视图3 2&#xff0c;在Xaml中添加Listview控件&#xff0c;Combobox控件。 <Grid ><Grid.RowDefinitions><RowDefinition Height"143*"/><RowDefinition Height"30"/>&l…

Leetcode经典题目之用队列实现栈

P. S.&#xff1a;以下代码均在VS2019环境下测试&#xff0c;不代表所有编译器均可通过。 P. S.&#xff1a;测试代码均未展示头文件stdio.h的声明&#xff0c;使用时请自行添加。 目录 1、题目展示2、题目分析3、完整代码演示4、结语 1、题目展示 前面我们了解过如何实现队列…

第五百回 Get路由管理

文章目录 1. 概念介绍2. 使用方法2.1 普通路由2.2 命名路由 3. 示例代码4. 内容总结 我们在上一章回中介绍了"使用get显示Dialog"相关的内容&#xff0c;本章回中将介绍使用get进行路由管理.闲话休提&#xff0c;让我们一起Talk Flutter吧。 1. 概念介绍 我们在本章…

掌握MySQL常用的命令

前言 MySQL是一个流行的开源关系型数据库管理系统&#xff0c;广泛应用于各种应用场景。熟练掌握MySQL的常用命令&#xff0c;对于数据库管理员和开发人员来说至关重要。本文将介绍MySQL数据库的一些基础术语、SQL语言分类&#xff0c;以及DDL、DML、DQL和DCL等操作&#xff0…