C语言如何使用断言避免踩坑

举报
小麦大叔 发表于 2021/12/03 23:18:04 2021/12/03
【摘要】 何为断言 断言一般是用于检测在某个程序位置程序必须满足某些条件的宏。一般用的多的可以分两种种情况: 前置条件:在某个程度点开始的地方后置条件:在某段程序执行结束后,一般用于检测执行结果 断言发生表示程序中存在错误。因此,断言是提高程序可靠性的有效手段。也是开发阶段快速定位问题的一种很好防御式编程方法。 在C语言中,断言是...

何为断言

断言一般是用于检测在某个程序位置程序必须满足某些条件的宏。一般用的多的可以分两种种情况:

  • 前置条件:在某个程度点开始的地方

  • 后置条件:在某段程序执行结束后,一般用于检测执行结果

断言发生表示程序中存在错误。因此,断言是提高程序可靠性的有效手段。也是开发阶段快速定位问题的一种很好防御式编程方法。

在C语言中,断言是一些条件判断的宏。比如C语言内置断言是用标准的 assert 宏实现的。当宏执行时,assert 的参数必须为真,否则程序中止并打印错误消息。

比如,在IAR中:

#define assert(test) ((test) ? (void) 0 : abort())
  

也可以编程者自己定义,比如:

#define assert(arg) { if( !(arg) ) { printf("assert in File="__FILE__" Line=%d ",__LINE__); return; } }
  

该怎么用

前置条件

比如某一个函数代码:


   
  1. #define ALLOWED_SIZE  (1024)
  2. int func(int size, char *buffer ) 
  3. {
  4.   assert( size <= ALLOWED_SIZE );
  5.   assert( format != NULL );
  6.   ...
  7. }

这个函数里,使用了两次断言判断函数执行的前置条件:

  • size必须要不大于ALLOWED_SIZE,func函数才真正执行其任务。因此,如果输入的size超过1024,func不会做任何处理。

  • buffer传入的地址必须不是NULL,否则func函数不会执行。

具体断言判断失败了,断言宏干了什么,需要看看这个宏的实现,有可能是直接返回,有可能整个程序直接终止执行。所以看看其实现就知道了。

后置条件

后置条件断言一般是指判断函数的执行结果。比如:


   
  1. int func(int size, char *buffer ) 
  2. {
  3.  int result;
  4.  
  5.  /*中间处理部分更新这个返回值*/
  6.   ...
  7.   
  8.   assert( result <= ALLOWED_SIZE );
  9.   return result;
  10. }

这样写表示这个函数的返回值永远不会大于ALLOWED_SIZE。如果大于了,就证明产生错误了。

什么时候用

断言的最常用和最有效的用途是检查前置条件——即指定和检查函数的输入条件。两个非常常见的用途:

  • 指针不是 NULL。

  • 索引和边界范围值是在设计的合理范围之类。

尤其如果写一个代码包给其他的人调用的时候,这样处理会使代码提高健壮性,易用性。

当代码调用带有前置条件的断言时,必须要确保满足该函数的前置条件。但这并不意味着必须断言检查调用的每个函数的参数!

调试的便利

  • 如果在程序测试和调试期间违反了前置条件,也就是说断言异常了,则调用包含前置条件的函数的代码中存在bug。

  • 如果在程序测试和调试期间违反了后置条件,则该断言前面部分代码可能有bug。

这样利用断言的打印,或者检测到断言指定的行为,就可以很快速的发现bug,而避免要在后期反复测试才能识别出bug。

那么什么时候用?首先,区分程序错误和运行时错误很重要:

  • 程序错误是一个bug,永远不应该发生。

  • 运行时错误可能在程序执行期间的任何时间发生。

断言不是处理运行时错误的机制。例如,由于用户在预期为正数时无意中输入了负数而导致的断言异常就是程序设计不合理。像这样的情况必须通过适当的错误检查和恢复代码(比如弹出一个提示输入合理范围)来处理,而不是通过断言来处置。

当然,实际是程序都可能会有bug,这些bug会在运行时出现。确切地说,断言要检查什么条件以及运行时错误检查代码要检查什么是设计问题。

如前所说,断言在可重用库中非常有效。比如在QT中:


   
  1. int main(int argc, char *argv[])
  2. {
  3.     QVector <int> list;
  4.     list.append(0);
  5.     list.append(1);
  6.     qDebug() << list.at(2);
  7.     
  8.     return 0;
  9. }

一运行,就会有这样的结果:


   
  1. ASSERT failure in QVector<T>::at: "index out of range", file C:\Qt\Qt5.7.1\5.7\mingw53_32\include/QtCore/qvector.h, line 429
  2. assert in File=..\src\main.cpp Line=4

因为list只有两个元素,list.at(2)则是去访问第3个,显然访问的元素不存在,所以就断言了。

—— The End ——

推荐好文  点击蓝色字体即可跳转

 到底什么是串级PID?

☞ 简易PID算法的快速扫盲49ee0ef22b248113f2bd08ab71dd59ac.gif

☞ 三面大疆惨败,因为不懂PID的积分抗饱和

☞ 6步!教你写一个mqtt调试助手

欢迎转发、留言、点赞、分享给你的朋友,感谢您的支持!

点击上方名片关注公众号

分享 💬  点赞 👍  在看 ❤️ 

以“三连”行动支持优质内容!

文章来源: great.blog.csdn.net,作者:小麦大叔,版权归原作者所有,如需转载,请联系作者。

原文链接:great.blog.csdn.net/article/details/120964102

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。