带参数的宏:
例如:/* 带参宏,可以像内联函数一样使用 */
#define LED1(a) if (a) \
GPIO_SetBits(LED1_GPIO_PORT,LED1_PIN); \
else \
GPIO_ResetBits(LED1_GPIO_PORT,LED1_PIN)
(前面每行后面都有一个斜杠,后面的斜杠是C语言里面的续行符,因为宏定义只对本行有效,所以使用了续航符)
防止头文件重复包含时使用宏定义:
例如:led.h的内容如下
#ifndef __LED_H
#define __LED_H
/*此处省略头文件的具体内容*/
#endif /* end of __LED_H */
解释如下:
在头文件的开头,使用“#ifndef”关键字,判断标号“__LED_H”是否被定义,若没
有被定义,则从“#ifndef”至“#endif”关键字之间的内容都有效,也就是说,这个头文件
若被其它文件“#include”,它就会被包含到其该文件中了,且头文件中紧接着使用
“#define”关键字定义上面判断的标号“__LED_H”。当这个头文件被同一个文件第二次
“#include”包含的时候,由于有了第一次包含中的“#define __LED_H”定义,这时再判
断“#ifndef __LED_H”,判断的结果就是假了,从“#ifndef”至“#endif”之间的内容都
无效,从而防止了同一个头文件被包含多次,编译时就不会出现“redefine(重复定义)”
的错误了。
一般来说,我们不会直接在 C 的源文件写两个“#include”来包含同一个头文件,但
可能因为头文件内部的包含导致重复,这种代码主要是避免这样的问题。如“bsp_led.h”
文件中使用了“#include ―stm32f4xx.h‖ ”语句,按习惯,可能我们写主程序的时候会在
main 文件写“#include ―bsp_led.h‖ 及#include ―stm32f4xx.h‖”,这个时候“stm32f4xx.h”文
件就被包含两次了,如果没有这种机制,就会出错。
至于为什么要用两个下划线来定义“__LED_H”标号,其实这只是防止它与其它普通
宏定义重复了,如我们用“ GPIO_PIN_0”来代替这个判断标号,就会因为 stm32f4xx.h 已
经定义了 GPIO_PIN_0,结果导致“bsp_led.h”文件无效了,“bsp_led.h”文件一次都没被
包含。
本文摘抄自野火老师的书: