0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

分享一下关于if和assert的内容

strongerHuang 来源:IOT物联网小镇 作者:IOT物联网小镇 2021-03-09 10:59 次阅读

嵌入式软件开发,实际项目通常会考虑代码的一些安全性和健壮性,一般来说,需要添加一些“检测”的代码。

下面就来为大家分享一下关于if和assert的内容。

一、前言

我们在撸代码的时候,经常需要对代码的安全性进行检查,例如:

1. 指针是否为空?

2. 被除数是否为 0?

3. 函数调用的返回结果是否有效?

4. 打开一个文件是否成功?

对这一类的边界条件进行检查的手段,一般都是使用 if 或者 assert 断言,无论使用哪一个,都可以达到检查的目的。那么是否就意味着:这两者可以随便使用,想起来哪个就用哪个?

这篇小短文我们就来掰扯掰扯:在不同的场景下,到底是应该用 if,还是应该使用 assert 断言?

写这篇文章的时候,我想起了孔乙己老先生的那个问题:茴香豆的“茴”字有几种写法?

似乎我们没有必要来纠结应该怎么选择,因为都能够实现想要的功能。以前我也是这么想的,但是,现在我不这么认为。

成为技术大牛、拿到更好的offer,也许就在这些细微之间就分出了胜负。

二、assert 断言

刚才,我问了下旁边的一位工作 5 年多的嵌入式开发者:if 和 assert 如何选择?他说:assert 是干什么的?!

看来,有必要先简单说一下 assert 断言。

assert() 的原型是:

void assert(int expression);

1. 如果宏的参数求值结果为非零值,则不做任何操作(no action);

2. 如果宏的参数是零值,就打印诊断消息,然后调用abort()。

例如下面的代码:

#include 《assert.h》int my_div(int a, int b){ assert(0 != b); return a / b;}

1. 当 b 不为 0 时,assert 断言什么都不做,程序往下执行;

2. 当 b 为 0 时,assert 断言就打印错误信息,然后终止程序;

从功能上来说,assert(0 != b); 与下面的代码等价:

if (0 == b){ fprintf(stderr, “b is zero.。.”); abort();}

assert 是一个宏,不是一个函数

在 assert.h 头文件中,有如下定义:

#ifdef NDEBUG #define assert(condition) ((void)0)#else #define assert(condition) /*implementation defined*/#endif

既然是宏定义,说明是在预处理的时候进行宏替换。(关于宏的更多内容,可以看一下这篇文章:提高代码逼格的利器:宏定义-从入门到放弃)。

从上面的定义中可以看到:

如果定义了宏 NDEBUG,那么 assert() 宏将不做什么动作,也就是相当于一条空语句:(void)0;,当在 release 阶段编译代码的时候,都会在编译选项中(Makefile)定义这个宏。

如果没有定义宏 NDEBUG,那么 assert() 宏将会把一些检查代码进行替换,我们在开发阶段执行 debug 模式编译时,一般都会屏蔽掉这 NDEBUG 这个宏。

三、if VS assert

还是以一个代码片段来描述问题,以场景化来讨论比较容易理解。

// brief: 把两个短字符串拼接成一个字符串char *my_concat(char *str1, char *str2){ int len1 = strlen(str1); int len2 = strlen(str2); int len3 = len1 + len2; char *new_str = (char *)malloc(len3 + 1); memset(new_str, 0 len3 + 1); sprintf(new_str, “%s%s”, str1, str2); return new_str;}

如果一个开发人员写出上面的代码,一定会被领导约谈的!它存在下面这些问题:

没有对输入参数进行有效性检查;

没有对 malloc 的结果进行检查;

sprintf 的效率很低;

。..

1. 使用 if 语句来检查

char *my_concat(char *str1, char *str2){ if (!str1 || !str2) // 参数错误 return NULL; int len1 = strlen(str1); int len2 = strlen(str2); int len3 = len1 + len2; char *new_str = (char *)malloc(len3 + 1); if (!new_str) // 申请堆空间失败 return NULL; memset(new_str, 0 len3 + 1); sprintf(new_str, “%s%s”, str1, str2); return new_str;}

2. 使用 assert 断言来检查

char *my_concat(char *str1, char *str2){ // 确保参数正确 assert(NULL != str1); assert(NULL != str2); int len1 = strlen(str1); int len2 = strlen(str2); int len3 = len1 + len2; char *new_str = (char *)malloc(len3 + 1); // 确保申请堆空间成功 assert(NULL != new_str); memset(new_str, 0 len3 + 1); sprintf(new_str, “%s%s”, str1, str2); return new_str;}

3. 你喜欢哪一个?

首先声明一点:以上这 2 种检查方式,在实际的代码中都很常见,从功能上来说似乎也没有什么影响。因此,没有严格的错与对之分,很多都是依赖于每个人的偏好习惯不同而已。

(1) assert 支持者

我作为 my_concat() 函数的实现者,目的是拼接字符串,那么传入的参数必须是合法有效的,调用者需要负责这件事。如果传入的参数无效,我会表示十分的惊讶!怎么办:崩溃给你看!

(2)if 支持者

我写的 my_concat() 函数十分的健壮,我就预料到调用者会乱搞,故意的传入一些无效参数,来测试我的编码水平。没事,来吧,我可以处理任何情况!

这两个派别的理由似乎都很充足!那究竟该如何选择?难道真的的跟着感觉走吗?

假设我们严格按照常规的流程去开发一个项目:

1. 在开发阶段,编译选项中不定义 NDEBUG 这个宏,那么 assert 就发挥作用;

2. 项目发布时,编译选项中定义了 NDEBUG 换个宏,那么 assert 就相当于空语句;

也就是说,只有在 debug 开发阶段,用 assert 断言才能够正确的检查到参数无效。而到了 release 阶段,assert 不起作用,如果调用者传递了无效参数,那么程序只有崩溃的命运了。

这说明什么问题?是代码中存在 bug?还是代码写的不够健壮?

从我个人的理解上看,这压根就是单元测试没有写好,没有测出来参数无效的这个 case!

4. assert 的本质

assert 就是为了验证有效性,它最大作用就是:在开发阶段,让我们的程序尽可能地 crash。每一次的 crash,都意味着代码中存在着 bug,需要我们去修正。

当我们写下一个 assert 断言的时候,就说明:断言失败的这种情况是不可以的,是不被允许的。必须保证断言成功,程序才能继续往下执行。

5. if-else 的本质

if-else 语句用于逻辑处理,它是为了处理各种可能出现的情况。就是说:每一个分支都是合理的,是允许出现的,我们都要对这些分支进行处理。

6. 我喜欢的版本

char *my_concat(char *str1, char *str2){ // 参数必须有效 assert(NULL != str1); assert(NULL != str2); int len1 = strlen(str1); int len2 = strlen(str2); int len3 = len1 + len2; char *new_str = (char *)malloc(len3 + 1); // 申请堆空间失败的情况,是可能的,是允许出现的情况。 if (!new_str) return NULL; memset(new_str, 0 len3 + 1); sprintf(new_str, “%s%s”, str1, str2); return new_str;}

对于参数而言:我认为传入的参数必须是有效的,如果出现了无效参数,说明代码中存在 bug,不允许出现这样的情况,必须解决掉。

对于资源分配结果(malloc 函数)而言:我认为资源分配失败是合理的,是有可能的,是允许出现的,而且我也对这个情况进行了处理。

当然了,并不是说对参数检查就要使用 assert,主要是根据不同的场景、语义来判断。例如下面的这个例子:

int g_state;void get_error_str(bool flag){ if (TRUE == flag) { g_state = 1; assert(1 == g_state); // 确保赋值成功 } else { g_state = 0; assert(0 == g_state); // 确保赋值成功 }}

flag 参数代表不同的分支情况,而赋值给 g_state 之后,必须保证赋值结果的正确性,因此使用 assert 断言。

五、总结

这篇文章分析了 C语言中比较晦涩、模糊的一个概念,似乎有点虚无缥缈,但是的确又需要我们停下来仔细考虑一下。

如果有些场景,实在拿捏不好,我就会问自己一个问题:

这种情况是否被允许出现?

不允许:就用 assert 断言,在开发阶段就尽量找出所有的错误情况;

允许:就用 if-else,说明这是一个合理的逻辑,需要进行下一步处理。

责任编辑:lq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • C语言
    +关注

    关注

    180

    文章

    7597

    浏览量

    136120
  • 代码
    +关注

    关注

    30

    文章

    4741

    浏览量

    68324
  • ASSERT
    +关注

    关注

    0

    文章

    17

    浏览量

    7228

原文标题:代码安全性和健壮性:如何在if和assert中做选择?

文章出处:【微信号:strongerHuang,微信公众号:strongerHuang】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    关于滤波器的个问题

    想问一下关于滤波器的个问题 http://www.gz.wiki/http://www.gz.wiki/
    发表于 11-12 21:43

    求助一下关于LLC谐振电流震荡的问题!

    LLC输出电压稳定,带载能力正常,但是谐振电流震荡很严重,调整回路也没有很好的改善,求助一下大佬,能帮忙看看吗!
    发表于 11-05 16:38

    求助,关于PFFB的设计问题,这样的设计会有什么好处和坏处?

    请教一下关于PFFB的设计问题,这样的设计会有什么好处和坏处?烦请解答谢谢
    发表于 10-16 06:52

    运放构成的电压跟随器的输出阻抗般如何确认大小?

    我想了解一下关于运放构成的电压跟随器的输出阻抗般如何确认大小。比如说由TL081构成的跟随器的输出阻抗。谢谢!
    发表于 08-16 06:19

    关于DAQExpress软件中电压的最大值最小值是根据什么来确定的?

    你好,我想问一下关于DAQExpress软件中电压的最大值最小值是根据什么来确定的?还有采样频率
    发表于 08-08 02:37

    关于AMC1301输入端口的问题求解

    我想请教一下关于AMC1301输入端口的问题,通过查看数据手册,输入端口电压范围是±250mv,我所做的设计需要进行母线电压采样,我给的电压是60V,经过分压后得到的电压是250mv,但是与AMC1301VINP连接后,电压就不是250mv了,这是什么原因,谢谢。
    发表于 07-31 07:49

    欢创播报 支付宝“碰一下”正式发布

    1 支付宝“碰一下”正式发布 近日,在支付宝开放日上,支付宝宣布升级条码支付体验,推出“支付宝碰一下”,用户无需展示付款码,解锁手机碰一下商家收款设备,最快步完成支付。据介绍,“碰
    的头像 发表于 07-11 11:32 848次阅读
    欢创播报  支付宝“碰<b class='flag-5'>一下</b>”正式发布

    STM8S007老是参数检查错误进入assert_failed()的原因?

    请哪位大侠解答一下,在使用STM8S007和STM8S105K4的时候,经过跟踪发现,调用些固件库函数时老是会出现参数检查错误而进入到assert_failed()函数,关键是参数都是库中预定义的
    发表于 05-17 06:18

    如何在if和assert中做选择

    的?! 看来,有必要先简单说一下 assert 断言。 assert() 的原型是: void assert(int expression); 1. 如果宏的参数求值结果为非零
    发表于 04-08 06:13

    aducm360在100Hz采样率的不动码可以达到多少位?

    请各位大牛帮忙回答一下关于aducm360的问题: 1. 在100Hz采样率的不动码可以达到多少位? 2. 在100Hz采样率的AD功耗是多少? 在线等,不甚感激!
    发表于 01-15 06:42

    盘点一下高通CES 2024汽车创新技术

    在CES2024上,我们看到英特尔和AMD加入,加上原来的英伟达,高通需要和这些跨行的对手在个赛道卷,目前高通在数字座舱、云连接、人工智能和自动驾驶领域是有定的积累的,我们来盘点一下高通本次的
    的头像 发表于 01-13 15:22 1751次阅读
    盘点<b class='flag-5'>一下</b>高通CES 2024汽车创新技术

    辐射测试,在近场要低于多少值?

    本文简单的介绍了一下关于辐射测试的些小知识点,有兴趣的朋友可以点进来了解一下
    的头像 发表于 01-04 11:23 1053次阅读
    辐射测试,在近场要低于多少值?

    分享下关于涂层附着力的原理和影响因素

    本期和大家分享下关于涂层附着力的原理和影响因素,首先我们都知道涂层和基材之间是可以通过机械咬合、物理吸附、形成氢键和化学键、互相扩散等作用结合在起,由于这些作用产生的附着力,决定了涂层与基材的附着力。
    的头像 发表于 12-14 10:05 2016次阅读

    介绍一下芯片的VIA pillar

    Via pillar,又可以叫Via ladder。貌似Cadence家喜欢叫pillar,synopsis喜欢叫ladder,我也不知道它们为啥不能统一一下名称。
    的头像 发表于 12-06 14:00 788次阅读

    物联网中的MQTT协议概述

    MQTT协议已经成了物联网通信中的个热词,今天我们起来学习一下关于MQTT协议的些知识。
    的头像 发表于 11-25 14:09 1273次阅读