assert使用验证

assert() 使用验证

在看项目源码看到一句话,用于入口参数检查,就想着自己能否在项目中也使用到:

assert(op >= sizeof(g_opStruct) / sizeof(g_opStruct[0]));

首先对于assert建立了解,我看的是这篇文章:assert() 怎么用?

  • 它是一个宏,接受一个整形表达式参数,当该表达式为假的时候,会打印出错误信息,并使程序停止运行。

  • 错误信息包含:所在的文件名 以及 该文件下的行号,可以帮助定位问题。

  • 它是一种异常处理方式,在类型检查时用得比较多,比如:除数为0, 函数指针为空,数组越界,参数检查。

  • 在出现这种会导致程序跑飞/崩溃/异常时,一条打印信息就从漫天找错 迅速定位到Bug原因。

  • 它定义在assert.h 中,而且很方便的一点是:一个宏定义#define NDEBUG 放在#include <assert.h> 前,就可以关闭该机制 从而减小程序空间占用。

我的验证使用

#include <stdio.h>
#include <stdint.h>
// #define NDEBUG
#include <assert.h>


typedef void (*menu_op_func)(uint8_t);
typedef struct OP_STRUCT
{
    int op_menu ; 				/*操作菜单*/
    menu_op_func opfun ;		/*带参数的操作方法*/
} OP_MENU_PAGE;

/*当前菜单*/
typedef enum
{
    MAIN_PAGE = 0,
    LOG_PAGE,
    CONF_PAGE,
    PARA_PAGE,
    VERSION_PAGE,
    PASSWD_PAGE,
    PASSWD_INPUT_PAGE,
    DATETIME_PAGE,
    RECOVERY_PAGE,
} OP_PAGE;

/*菜单操作表定义*/
static OP_MENU_PAGE g_opStruct[] =
{
    {MAIN_PAGE      	, 	NULL},
    {LOG_PAGE					, 	NULL},
    {CONF_PAGE      	, 	NULL},
    {PARA_PAGE				,   NULL},
    {VERSION_PAGE   	, 	NULL},
    {PASSWD_PAGE			,   NULL},
    {PASSWD_INPUT_PAGE,		NULL},
    {DATETIME_PAGE		,		NULL},
    {RECOVERY_PAGE		,   NULL},
};

/*跳转到表所对应的页面*/
static int JUMP_Table(int8_t op, uint8_t KeyValue)
{
    assert(op >= sizeof(g_opStruct) / sizeof(g_opStruct[0]));
    assert(op < 0);
    printf("parameter is ok! \n");
    return 0 ;
}


int main(void)
{
    uint8_t test_var_i = 5;
    uint8_t test_var_j = 10;

    JUMP_Table(test_var_i, 24);
    JUMP_Table(test_var_j, 23);
    return 0;
}

image-20220108205639280

我的发现

通过验证使用发现和想象中有点儿不大一样:

第一次参数调用时传入的数值为5,满足条件检查,此时应输出一条:

parameter is ok!

但实际情况是没有输出,直接输出的是:

Assertion failed: op >= sizeof(g_opStruct) / sizeof(g_opStruct[0]), file test1.c, line 45

后来才意识到:那个源码是有问题的:用法错了,表达式为假时输出,而非为真时。

通过关闭assert机制,观察到的现象是:

parameter is ok!
parameter is ok!

意识到:那个源码是有问题的:用法错了,表达式为假时输出,而非为真时。

通过关闭assert机制,观察到的现象是:

parameter is ok!
parameter is ok!


版权声明:本文为quanquanxiaobu原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。