来源:公众号(c语言与cpp编程)
断言assert原型
void assert(int expression);
assert宏的原型定义在<assert.h>中,其作用是先计算表达式expression的值为假(即为0),那么它就先向stderr打印一条出错信息,然后通过条用abort来终止程序;
使用assert的缺点是,频繁的调用会极大的影响程序的性能,增加额外的开销。
在调试结束后,可以通过在包含#include 的语句之前插入 #define NDEBUG 来禁用assert调用,示例代码如下:
代码语言:javascript复制1 #include
2 #define NDEBUG
3 #include
断言assert使用规则
代码语言:javascript复制
#include <stdio.h>
#include <assert.h>
int main(void)
{
int i;
i = 1;
assert(i );
printf(“%dn”,i);
return 0;
}
看运行结果,如果给定的i初始值为1,所以其运行结果不会为错,如下图所示
很显然是2,不会出错
如果将i初始值改成0,那么就会出现如下错误:
出现异常
上面这个错误是很典型异常,可以考虑用assert排查。
根据提示我们很快就能定位到错误点,就在assert(i )处;既然assert这么便于定位出错点,在工程中使用它就显得很有必要;但其也有一定的使用规则;
断言语句不会永远被执行,可以屏蔽也可以启用,这就要求assert不管是在屏蔽还是启用状态下都不能对我们本身代码有所影响,这样刚才我们在代码中使用的assert(i )
就不行,因为如果禁用了assert
,那i 就不能执行;正确的做法应该是:assert(i);i ;
那么我们一般在什么情况下使用断言呢?
主要体现在以下几个方面:
1. 可以在预计正常情况下程序不会到达的地方放置断言。(如assert(0);)
2. 使用断言测试方法的前置条件和后置条件;
- 前置条件:代码执行前必须具备的特性;
- 后置条件:代码执行后必须具备的特性;
3. 使用断言检测类的不变状态,确保任何情况下,某个变量的状态或范围必须满足。
断言assert使用规则
当然我们在使用断言的过程中会有一些我们应该注意的事项和养成一些良好的习惯,如:
1. 每个assert只检验一个条件,因为同时检验多个条件时,如果断言失败,我们就无法直观的判断哪个条件失败;
无法直观的判断哪个条件失败:
代码语言:javascript复制assert(nOffset>=0 && nOffset nSize<=m_nInfomationSize);
只检验一个条件,比较直观:
代码语言:javascript复制assert(nOffset >= 0);
assert(nOffset nSize <= m_nInfomationSize);
- 不能使用改变环境的语句,就像我们上面的代码改变了i变量,在实际编写代码的过程中是不能这样做的;
例如:
代码语言:javascript复制assert(i < 100)
不好:这是因为如果出错,比如在执行之前i=100,那么这条语句就不会执行,那么i 这条命令就没有执行。
代码语言:javascript复制assert(i < 100)
i ;
正确。
3. assert和后面的语句应该空一行,以形成逻辑和视觉上的一致性,也算是一种良好的编程习惯,让编写的代码有一种视觉上的美感;
4. 有的地方,assert不能代替条件过滤;
程序一般分为Debug 版本和Release 版本,Debug 版本用于内部调试,Release 版本发行给用户使用。断言assert 是仅在Debug 版本起作用的宏,它用于检查"不应该"发生的情况。
5. 放在函数参数的入口处检查传入参数的合法性;
代码语言:javascript复制int resetBufferSize(int nNewSize)
{
//功能:改变缓冲区大小,
//参数:nNewSize 缓冲区新长度
//返回值:缓冲区当前长度
//说明:保持原信息内容不变 nNewSize<=0表示清除缓冲区
assert(nNewSize >= 0);
assert(nNewSize <= MAX_BUFFER_SIZE);
...
}
在我们使用C语言/C 做工程项目时,如果我们能在代码中合理的使用assert,能使我们创建更稳定、质量更好且不易于出错的代码;当需要在一个值为FALSE时中断当前操作的话就可以使用断言。
单元测试必须使用断言;另外除了类型检查和单元测试外,断言还提供了一种确定各种特性是否在程序中得到维护的极好的方法;