当前位置:首页 > > 嵌入式微处理器
[导读]在我们阅读一些大型的代码或者库的时候,一般都会看到有#error和#warning,可能有些小伙伴一扫而过,并没有了解清楚这些预编译指令到底该怎么用,写了很久的代码估计也重来没有敲过他们。

1、情景再现


咬金,C语言学得怎么样了?

我都敲了2年C代码了,还有我不知道的吗?

你狂,你继续狂!那我问你#error,#warning,#def这三个预编译干啥的?

额~~~,这就尴尬了,平时看代码基本上都见过,没怎么研究过哦。

哈哈~,要不要我教教你呀?

小鲁班,跟大哥说说呗!

行吧,待会说我这队友啥也不懂!

1

#error 与#warning


    谈到预编译大家常用的有#if、#else、#ifdef、#ifndef、#endif等等条件编译选项。

    

    然而在我们阅读一些大型的代码或者库的时候,一般都会看到有#error和#warning,可能有些小伙伴一扫而过并没有了解清楚这些预编译指令到底该怎么用,写了很久的代码估计也重来没有敲过他们。


  • #error / #warning

  • 形式 : #error / #warning message 

  • 用 : 生成一个编译错误事件并停止编译/发出警告信息

  • 注意 : message 可以不需要双引号。


参考demo:
#include 
#include 

//#define configUART_N 5

#ifndef configUART_N
    #error configUART_N must define
// #error "configUART must define"
// #warning "configUART must define"
#endif

#if configUART_N > 4
   #error configUART_N must not be less than 4
// #error "configUART_N must not be less than 4"
// #warning "configUART_N must not be less than 4"
#endif

/***************************************
 * Fuction: 进行预编译测试 
 * Author :(最后一个bug) 
 **************************************/

int main(int argc, char *argv[]) {
    printf("公众号;最后一个bug\n");
    return 0;
}
输出结果:


  • 编译失败,无法生成可执行文件


  • 上面是放开宏,且使用warning的情况,无其他错误的情况下可以生成可执行文件。


解释一下:
  • 通过上面的测试代码可以了解到,通过配合条件预编译#if等,#error和#warning能够在编译过程中分别以错误和告警的形式提醒开发人员注意相关代码设计问题,从而保证代码正确性。


  • 这样对于发布一些庞大的库代码时,为了让开发人员正确的使用库,这些提示会帮助他更好的移植代码。


  • 那么经常有很多小伙伴编译出来的代码有一大堆warning,总是觉得warning关系不大,然而warning也是分不同类型的,对于一些未使用的变量倒关系不大,其他情况还是要认真对待,最好是做到"0 Error,0 warning".



2

#undef


   #undef标识符用于把前面的宏定义名取消,别看这宏用得不多,作用可大着呢,下面我简单举几个例子:


1

局部宏定义 


    一旦定义了宏,那么该文件中往下所有的代码都可以使用该宏,即使是函数内部,这样导致宏比较混乱,如下面代码:

参考demo:
#include 
#include 

#define configRatio 10

/***************************************
 * Fuction: 获得传感器电压值 
 * Author :(最后一个bug) 
 **************************************/

int GetSensorVolt(void)
{    
#define configRatio 1
    int ret = 0;
     ret = configRatio*1024//比例因子*AD值 

    return ret;
//#undef configRatio
}

/***************************************
 * Fuction: 获得传感器电压值 
 * Author :(最后一个bug) 
 **************************************/

int GetSensorCurr(void)
{    
#define configRatio 2
    int ret = 0;
     ret = configRatio*1024//比例因子*AD值 

    return ret;
//#undef configRatio
}

/***************************************
 * Fuction: 进行预编译测试 
 * Author :(最后一个bug) 
 **************************************/

int main(int argc, char *argv[]) {

    printf("configRatio = %d\n",configRatio);  //报宏未定义 
    printf("GetSensorVolt = %d\n",GetSensorVolt());  
    printf("GetSensorCurr = %d\n",GetSensorCurr()); 
    printf("公众号;最后一个bug\n");
    return 0;
}
输出结果:


解释一下:
  • 假如我们没有注意到函数内部的同名宏定义,当然告警也没管,那么在main函数中使用同名宏定义就可能不是我们期待的最上面的宏定义,造成程序bug。


  • 所以我们可以使用#undef来限制每个宏的作用域,如果每个函数内部都使用了#undef,那么main函数中再使用会报宏没有定义,这样便可以找到问题,当然也可以通过警告了解到。


2

选择接口


   通过宏来切换不同的接口供程序使用:

参考demo:
#include 
#include 
#include 


#define DEV_SPI
#include "Drive.h"
#undef DEV_SPI
/***************************************
 * Fuction: 进行预编译测试 
 * Author :(最后一个bug) 
 **************************************/

int main(int argc, char *argv[]) {

    char *strbug = "the last bug" ;

    SendData(strbug);
    ProcessData(strbug);
    printf("公众号;最后一个bug\n");
    return 0;
}


#include 


#ifdef DEV_UART

#define SendData(s)    printf("UART Send:%s\n",s)
#define ProcessData(s)  printf("UART Process:%s\n",s)

#endif

#ifdef DEV_CAN

#define SendData(s)    printf("CAN Send:%s\n",s)
#define ProcessData(s)  printf("CAN Process:%s\n",s)

#endif

#ifdef DEV_SPI

#define SendData(s)    printf("Spi Send:%s\n",s)
#define ProcessData(s)  printf("Spi Process:%s\n",s)

#endif
输出结果:



3

自定义接口


   当多个人维护一套代码的时候,有些同事喜欢调用库函数接口,而有些同事喜欢调用自定义接口,为了方便统一使用自定义接口或者库接口,我们会进行如下操作:

参考demo:
#include 
#include 
#include 

#include "Drive.h"
//#undef printf

/***************************************
 * Fuction: 进行预编译测试 
 * Author :(最后一个bug) 
 **************************************/

int main(int argc, char *argv[]) {

    char *strbug = "the last bug" ;

    printf("公众号;最后一个bug\n");
    return 0;
}


#ifndef __DRIVE_H__
#define __DRIVE_H__

#define  printf printf("please use Kprintf!\n");

extern void Kprintf(char *str);

#endif
输出结果:


  • 这样下面的代码你就只能够使用Kprintf来进行输出打印,而当我们放开注释掉的宏,这样就又可以使用printf了,还是比较方便的。



咬金,懂了没 ?

小鲁班,这些知识都被你学到了!666

2、结束语

    上面这几个比较"冷门"的知识认真想想其实还是挺有用的,可能现在的产品都急于快速上市,对于代码的雕琢还有所欠缺的,一份成熟的代码不仅仅只是稳定,还有后期的维护、扩展等等都是值得考虑的。


-END-


来源 | 最后一个bug

作者 | bug菌


整理文章为传播相关技术,版权归原作者所有 |

如有侵权,请联系删除 |


【1】知名半导体MCU大厂软件开发C代码规范

【2】工业项目,用MCU还是PLC?

【3】为什么嵌入式工程师会对8位MCU有误解?

【4】RGB 接口和 MCU 接口有什么不一样?

【5】8位微控制器(MCU)的隐形成本



免责声明:本文内容由21ic获得授权后发布,版权归原作者所有,本平台仅提供信息存储服务。文章仅代表作者个人观点,不代表本平台立场,如有问题,请联系我们,谢谢!

嵌入式ARM

扫描二维码,关注更多精彩内容

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭