子程序调用与宏定义的异同_【干货】这几个嵌入式C语言实用宏技巧你不能不知道……...

f07d86ee7b7d8575a399862af33091e3.png

文章字数:2000 干货指数:⭐⭐⭐⭐⭐

01宏打印函数

在我们的嵌入式开发中,使用printf打印一些信息是一种常用的调试手段。但是,在打印的信息量比较多的时候,就比较难知道哪些信息在哪个函数里进行打印。

特别是对于异常情况的打印,我们需要快速定位到异常情况的位置。

这时候我们可以使用宏定义来封装一个宏打印函数,这个宏打印函数可以显示打印信息所在的文件、行数、函数名等信息。如:

#define DBG_PRINTF(fmt, args...)  \
{\
 printf("<> ", __FILE__, __LINE__, __FUNCTION__);\
 printf(fmt, ##args);\
}

左右滑动查看全部代码>>>

使用范例:

115ef01d08131de129dae7123a6eef2f.png

029d6396980267b159eeef404677b78e.png

可见,使用方法与printf的使用方法一样,而且每条打印语句开头都会打印调试信息所在的文件名、行号、函数名信息,方便我们查找一些调试信息。

其中,__FILE__、__LINE__、__FUNCTION__这三个宏是编译器内置宏定义,分别代表调试信息所在文件、行号、函数。

除此之外,常用的宏还有:__DATE__、__TIME__,分别代表当前的编译日期与时间。如:

DBG_PRINTF("Compile Time: %s  %s\n", __DATE__, __TIME__);

dd22a6ed7c9517c1972c577fe4c2d9f7.png

第二条printf中的##符号是为了处理args不代表任何参数的情况。如:

DBG_PRINTF("Hello world");

当不加##符号是,以上宏的第二条语句被拓展为:

printf("Hello world\n", );

可见,多出了一个逗号,这个逗号是多余的。

加上##符号后,以上宏的第二条语句被拓展为:

printf("Hello world\n");

这才是我们想要的结果。其实这些结果我们通过查看预处理文件可以清晰的知道:

eb39b5a66566760397a6cce46bda93e3.png

b84dc04dae8d9a63bb313bbf88210658.png

235301b34de79ef89514dabdcc988be2.png

最后需要注意的是,这个DBG_PRINTF还是与printf不一样的。DBG_PRINTF宏是两条语句的组合,无返回值;而printf的原型是:

int printf (const char *__format, ...)

但是我们一般都很少使用printf的返回值,所以DBG_PRINTF的用法与printf函数基本一致。

02打印调试宏开关

通常情况下,一些打印调试信息只是在我们调试阶段需要的,在程序发布阶段是不需要的。

所以,为了避免打印调试信息带来的资源开销,我们可以把这些打印调试语句给注释掉。

一种方法是逐句进行注释,这是一种比较低效的方法。比较高效的方法就是添加调试宏开关,利用条件编译来选择打印/不打印调试信息。

比如我们可以把上面的代码改造为:

#define  DEBUG   1  

#if DEBUG
  #define DBG_PRINTF(fmt, args...)  \
  {\
    printf("<> ", __FILE__, __LINE__, __FUNCTION__);\
    printf(fmt, ##args);\
  }
#else
  #define DBG_PRINTF(fmt, args...)   
#endif

根据DEBUG宏的值来选择对应的打印宏函数。当DEBUG的值为1时启动相关的打印调试语句,DEBUG的值为0时则关闭打印调试语句。

这样我们就可以很方便的通过设置DEBUG宏的值来启动与关闭我们整个工程的DBG_PRINTF打印调试信息。

03do{}while(0) 其实,上面我们封装的打印宏DBG_PRINTF还有一点缺陷,比如我们与if、else使用的时候,会有这样的一种使用情况:

db27afcd402becef4cdd71a8248a7555.png

此时会报语法错误。为什么呢? 同样的,我们可以先来看一下我们的demo代码预处理过后,相应的宏代码会被转换为什么。如:

8914bf31d4cd5daddebd0e8261d89730.png

这里我们可以看到,我们的if、else结构代码被替换为如下形式:
if(c)
{ /* ....... */ };
else
{ /* ....... */ };
显然,出现了语法错误。if之后的大括号之后不能加分号,这里的分号其实可以看做一条空语句,这个空语句会把if与else给分隔开来,导致else不能正确匹配到if,导致语法错误。 为了解决这个问题,有几种方法。第一种方法是:把分号去掉。代码变成:

85755b89bf138ca288ef8565bb4272e9.png

第二种方法是:在if之后使用DBG_PRINTF打印调试时总是加{}。代码变成:

b39ee62505cdb576d7477aa1a427fb91.png

以上两种方法都可以正常编译、运行了。 但是,我们C语言中,每条语句往往以分号结尾;并且,总有些人习惯在if判断之后只有一条语句的情况下不加大括号;而且我们创建的DBG_PRINTF宏函数的目的就是为了对标printf函数,printf函数的使用加分号在任何地方的使用都是没有问题的。 基于这几个原因,我们有必要再对我们的DBG_PRINTF宏函数进行一个改造。 下面引入do{}while(0)来对我们的DBG_PRINTF进行一个简单的改造。改造后的DBG_PRINTF宏函数如下:
#define DBG_PRINTF(fmt, args...)  \
do\
{\
printf("<> ", __FILE__, __LINE__, __FUNCTION__);\
printf(fmt, ##args);\
}while(0)
这里的do...while循环的循环体只执行一次,与不加循环是效果一样。并且,可以避免了上面的问题。预处理文件:

5efa413b50e87fc908fa4050cb5323dd.png

我们的宏函数实体中,while(0)后面不加分号,在实际调用时补上分号,既符合了C语言语句分号结尾的习惯,也符合了do...while的语法规则。

使用do{}while(0)来封装宏函数可能会让很多初学者看着不习惯,但必须承认的是,这确确实实是一种很常用的方法。

在STM32的HAL库中搜索while(0):

42cbb703531c86decc17422d1068f62c.png

875d72187c2db5435eba5b529fff741a.png

在Linux源码中搜索while(0):

152c9273ea6ef205e377c06e0866f40f.png

可见,在实际应用中,do{}while(0)用的很多。

04#运算符与##运算符

这两个运算符之前也有分享过,这里顺便也提一下。

#号作为一个预处理运算符,可以把记号转换成字符串。

例如,如果A是一个宏形参,那么#A就是转换为字符串"A"的形参名。这个过程称为字符串化(stringizing)。以下程序演示这个过程:

48afe5ad9228b611966db772a4f8916d.png

a750db0c39fbe7a053920688ace2fa6a.png

##运算符可以把两个记号组合成一个记号。以下程序演示这个过程:

4a5d3621511881fdc41aae3905f4f446.png

ebfd18a9a7be8b4c7c68d1fc2ba75c68.png

这个运算符用得很多。如:

670d7d6c76336bcb26c48fdca9e285f1.png

872f2963bdc954b651cd8ac5ec1ece15.png

1

《13招!嵌入式 C 语言你不能不知道的优化技巧……》

2

《8000字!嵌入式 C 语言你必须知道的数据储存与程序编写知识!》

3

《20个嵌入式工程师常见面试题(内附答案)》

f321e9a086aab5892d3ed988efd51bad.png

cccecfc47aaf3cae2aff066f6a109efa.png

本周直播 | 点击查看?

d4b4a8f4de1303e16f10646574c421de.png

周一 | 基于STM32物联网项目开发

1、信息采集要如何实现?

2、Zigbee组网开发如何开展?

60d909ad178d66166f5cb1b80bd81008.png

周二 | 1小时搞定SPI通信原理及应用

1、SPI总线通信原理

2、SPI总线与IIC的异同

3、通信总线应用场景

905c36441e5baf33714c2f4b2400775b.png

 觉得有用点个在看 

d90319c46f912d08f8731948a0332b06.png
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值