#define 高级用法

本文详细介绍了C语言中宏定义的多种用法,包括如何定义可变参数宏、使用字符串化操作符#、符号连接操作符##等,并解释了这些宏在预处理阶段的行为。



一、LINUX C中用define定义可变参数的宏



一般在调试打印Debug信息的时候, 需要可变参数的宏. 从C99开始可以使编译器标准支持可变参数宏(variadic macros), 另外GCC也支持可变参数宏, 但是两种在细节上可能存在区别.
1. __VA_ARGS__
        __VA_ARGS__ 将 "..." 传递给宏 . 如
        #define debug(format, ...) fprintf(stderr, format, __VA_ARGS__)

2. GCC的复杂宏
        GCC使用一种不同的语法,从而可以给可变参数一个名字,如同其它参数一样.
        #define debug(format, args...) fprintf (stderr, format, args)
        这和第一条的宏例子是完全一样的,但是这么写可读性更强并且更容易进行描述.

3. ##__VA_ARGS__
        上面两个定义的宏,如果出现 debug("A Message")的时候,由于宏展开后有个多余的逗号,所以将导致编译错误.
        为了解决这个问题,CPP 使用一个特殊的"##"操作,格式如下:
        #define debug(format, ...) fprintf (stderr, format, ## __VA_ARGS__)
        这里,如果可变参数被忽略或为空,"##"操作将使预处理器(preprocessor)去除掉它前面的那个逗号.


上文来自:http://blog.chinaunix.net/uid-17240700-id-2813911.html







二、define宏定义中的#,##,@#及\符号


1、# (stringizing)字符串化操作符。

其作用是:将宏定义中的传入参数名转换成用一对双引号括起来参数名字符串。其只能用于有传入参数的宏定义中,且必须置于宏定义体中的参数名前。

如:

#define example(instr) printf("the input string is:\t%s\n",#instr)

#define example1(instr) #instr

当使用该宏定义时:

example(abc); 在编译时将会展开成:printf("the input string is:\t%s\n","abc");

string str=example1(abc); 将会展成:string str="abc";

注意:

对空格的处理

a。忽略传入参数名前面和后面的空格。

   如:str=example1(   abc ); 将会被扩展成 str="abc";

b.当传入参数名间存在空格时,编译器将会自动连接各个子字符串,用每个子字符串中只以一个空格连接,忽略其中多余一个的空格。

   如:str=exapme( abc    def); 将会被扩展成 str="abc def";

 2、## (token-pasting)符号连接操作符

宏定义中:参数名,即为形参,如#define sum(a,b) (a+b);中a和b均为某一参数的代表符号,即形式参数。

而##的作用则是将宏定义的多个形参成一个实际参数名。

如:

#define exampleNum(n) num##n

int num9=9;

使用:

int num=exampleNum(9); 将会扩展成 int num=num9;

注意:

1.当用##连接形参时,##前后的空格可有可无。

如:#define exampleNum(n) num ## n 相当于 #define exampleNum(n) num##n

2.连接后的实际参数名,必须为实际存在的参数名或是编译器已知的宏定义

// preprocessor_token_pasting.cpp

#include <stdio.h>

#define paster( n ) printf_s( "token" #n " = %d", token##n )

int token9 = 9;

 int main()

{

   paster(9);

}

运行结果:

token9 = 9

 3、@# (charizing)字符化操作符。

只能用于有传入参数的宏定义中,且必须置于宏定义体中的参数名前。作用是将传的单字符参数名转换成字符,以一对单引用括起来。

#define makechar(x)  #@x

a = makechar(b);

展开后变成了:

a= 'b';

 

4、\ 行继续操作符

当定义的宏不能用一行表达完整时,可以用"\"表示下一行继续此宏的定义。



上文来自:http://hi.baidu.com/lowen_pig/item/1163b2d530cbec826dce3f8a






附:编译器内置宏(补充于2014.03.31)

ANSI C标准中有几个标准预定义宏(也是常用的):

__LINE__:在源代码中插入当前源代码行号;

__FILE__:在源文件中插入当前源文件名;

__DATE__:在源文件中插入当前的编译日期

__TIME__:在源文件中插入当前编译时间;

__STDC__:当要求程序严格遵循ANSI C标准时该标识被赋值为1;

__cplusplus:当编写C++程序时该标识符被定义。










#define GPIOA0_PORT GPIOA #define GPIOA0_PIN GPIO_PIN_0 #define GPIOA0 GPIOA0_PORT,GPIOA0_PIN #define GPIOA1_PORT GPIOA #define GPIOA1_PIN GPIO_PIN_1 #define GPIOA1 GPIOA1_PORT,GPIOA1_PIN #define GPIOA2_PORT GPIOA #define GPIOA2_PIN GPIO_PIN_2 #define GPIOA2 GPIOA2_PORT,GPIOA2_PIN #define GPIOA3_PORT GPIOA #define GPIOA3_PIN GPIO_PIN_3 #define GPIOA3 GPIOA3_PORT,GPIOA3_PIN #define GPIOA4_PORT GPIOA #define GPIOA4_PIN GPIO_PIN_4 #define GPIOA4 GPIOA4_PORT,GPIOA4_PIN #define GPIOA5_PORT GPIOA #define GPIOA5_PIN GPIO_PIN_5 #define GPIOA5 GPIOA5_PORT,GPIOA5_PIN #define GPIOA6_PORT GPIOA #define GPIOA6_PIN GPIO_PIN_6 #define GPIOA6 GPIOA6_PORT,GPIOA6_PIN #define GPIOA7_PORT GPIOA #define GPIOA7_PIN GPIO_PIN_7 #define GPIOA7 GPIOA7_PORT,GPIOA7_PIN #define GPIOA8_PORT GPIOA #define GPIOA8_PIN GPIO_PIN_8 #define GPIOA8 GPIOA8_PORT,GPIOA8_PIN #define GPIOA9_PORT GPIOA #define GPIOA9_PIN GPIO_PIN_9 #define GPIOA9 GPIOA9_PORT,GPIOA9_PIN #define GPIOA10_PORT GPIOA #define GPIOA10_PIN GPIO_PIN_10 #define GPIOA10 GPIOA10_PORT,GPIOA10_PIN #define GPIOA11_PORT GPIOA #define GPIOA11_PIN GPIO_PIN_11 #define GPIOA11 GPIOA11_PORT,GPIOA11_PIN #define GPIOA12_PORT GPIOA #define GPIOA12_PIN GPIO_PIN_12 #define GPIOA12 GPIOA12_PORT,GPIOA12_PIN #define GPIOA13_PORT GPIOA #define GPIOA13_PIN GPIO_PIN_13 #define GPIOA13 GPIOA13_PORT,GPIOA13_PIN #define GPIOA14_PORT GPIOA #define GPIOA14_PIN GPIO_PIN_14 #define GPIOA14 GPIOA14_PORT,GPIOA14_PIN #define GPIOA15_PORT GPIOA #define GPIOA15_PIN GPIO_PIN_15 #define GPIOA15 GPIOA15_PORT,GPIOA15_PIN #define GPIOB0_PORT GPIOB #define GPIOB0_PIN GPIO_PIN_0 #define GPIOB0 GPIOB0_PORT,GPIOB0_PIN #define GPIOB1_PORT GPIOB #define GPIOB1_PIN GPIO_PIN_1 #define GPIOB1 GPIOB1_PORT,GPIOB1_PIN #define GPIOB2_PORT GPIOB #define GPIOB2_PIN GPIO_PIN_2 #define GPIOB2 GPIOB2_PORT,GPIOB2_PIN #define GPIOB3_PORT GPIOB #define GPIOB3_PIN GPIO_PIN_3 #define GPIOB3 GPIOB3_PORT,GPIOB3_PIN #define GPIOB4_PORT GPIOB #define GPIOB4_PIN GPIO_PIN_4 #define GPIOB4 GPIOB4_PORT,GPIOB4_PIN #define GPIOB5_PORT GPIOB #define GPIOB5_PIN GPIO_PIN_5 #define GPIOB5 GPIOB5_PORT,GPIOB5_PIN #define GPIOB6_PORT GPIOB #define GPIOB6_PIN GPIO_PIN_6 #define GPIOB6 GPIOB6_PORT,GPIOB6_PIN #define GPIOB7_PORT GPIOB #define GPIOB7_PIN GPIO_PIN_7 #define GPIOB7 GPIOB7_PORT,GPIOB7_PIN #define GPIOB8_PORT GPIOB #define GPIOB8_PIN GPIO_PIN_8 #define GPIOB8 GPIOB8_PORT,GPIOB8_PIN #define GPIOB9_PORT GPIOB #define GPIOB9_PIN GPIO_PIN_9 #define GPIOB9 GPIOB9_PORT,GPIOB9_PIN #define GPIOB10_PORT GPIOB #define GPIOB10_PIN GPIO_PIN_10 #define GPIOB10 GPIOB10_PORT,GPIOB10_PIN #define GPIOB11_PORT GPIOB #define GPIOB11_PIN GPIO_PIN_11 #define GPIOB11 GPIOB11_PORT,GPIOB11_PIN #define GPIOB12_PORT GPIOB #define GPIOB12_PIN GPIO_PIN_12 #define GPIOB12 GPIOB12_PORT,GPIOB12_PIN #define GPIOB13_PORT GPIOB #define GPIOB13_PIN GPIO_PIN_13 #define GPIOB13 GPIOB13_PORT,GPIOB13_PIN #define GPIOB14_PORT GPIOB #define GPIOB14_PIN GPIO_PIN_14 #define GPIOB14 GPIOB14_PORT,GPIOB14_PIN #define GPIOB15_PORT GPIOB #define GPIOB15_PIN GPIO_PIN_15 #define GPIOB15 GPIOB15_PORT,GPIOB15_PIN #define GPIOC0_PORT GPIOC #define GPIOC0_PIN GPIO_PIN_0 #define GPIOC0 GPIOC0_PORT,GPIOC0_PIN #define GPIOC1_PORT GPIOC #define GPIOC1_PIN GPIO_PIN_1 #define GPIOC1 GPIOC1_PORT,GPIOC1_PIN #define GPIOC2_PORT GPIOC #define GPIOC2_PIN GPIO_PIN_2 #define GPIOC2 GPIOC2_PORT,GPIOC2_PIN #define GPIOC3_PORT GPIOC #define GPIOC3_PIN GPIO_PIN_3 #define GPIOC3 GPIOC3_PORT,GPIOC3_PIN #define GPIOC4_PORT GPIOC #define GPIOC4_PIN GPIO_PIN_4 #define GPIOC4 GPIOC4_PORT,GPIOC4_PIN #define GPIOC5_PORT GPIOC #define GPIOC5_PIN GPIO_PIN_5 #define GPIOC5 GPIOC5_PORT,GPIOC5_PIN #define GPIOC6_PORT GPIOC #define GPIOC6_PIN GPIO_PIN_6 #define GPIOC6 GPIOC6_PORT,GPIOC6_PIN #define GPIOC7_PORT GPIOC #define GPIOC7_PIN GPIO_PIN_7 #define GPIOC7 GPIOC7_PORT,GPIOC7_PIN #define GPIOC8_PORT GPIOC #define GPIOC8_PIN GPIO_PIN_8 #define GPIOC8 GPIOC8_PORT,GPIOC8_PIN #define GPIOC9_PORT GPIOC #define GPIOC9_PIN GPIO_PIN_9 #define GPIOC9 GPIOC9_PORT,GPIOC9_PIN #define GPIOC10_PORT GPIOC #define GPIOC10_PIN GPIO_PIN_10 #define GPIOC10 GPIOC10_PORT,GPIOC10_PIN #define GPIOC11_PORT GPIOC #define GPIOC11_PIN GPIO_PIN_11 #define GPIOC11 GPIOC11_PORT,GPIOC11_PIN #define GPIOC12_PORT GPIOC #define GPIOC12_PIN GPIO_PIN_12 #define GPIOC12 GPIOC12_PORT,GPIOC12_PIN #define GPIOC13_PORT GPIOC #define GPIOC13_PIN GPIO_PIN_13 #define GPIOC13 GPIOC13_PORT,GPIOC13_PIN #define GPIOC14_PORT GPIOC #define GPIOC14_PIN GPIO_PIN_14 #define GPIOC14 GPIOC14_PORT,GPIOC14_PIN #define GPIOC15_PORT GPIOC #define GPIOC15_PIN GPIO_PIN_15 #define GPIOC15 GPIOC15_PORT,GPIOC15_PIN //debug uart #define _UART_DEBUG_ #ifdef _UART_DEBUG_ #define UART_DEBUG UART5 #define UART_DEBUG_GPIO GPIOB #define UART_DEBUG_APBxClkCmd RCC_EnableAPB2PeriphClk #define UART_DEBUG_GPIO_APBxClkCmd RCC_EnableAPB2PeriphClk #define UART_DEBUG_CLK RCC_APB2_PERIPH_UART5 #define UART_DEBUG_GPIO_CLK RCC_APB2_PERIPH_GPIOB #define UART_DEBUG_RxPin GPIO_PIN_5 #define UART_DEBUG_TxPin GPIO_PIN_4 #define UART_DEBUG_Rx_GPIO_AF GPIO_AF7_UART5 #define UART_DEBUG_Tx_GPIO_AF GPIO_AF6_UART5 #define UART_DEBUG_IRQn UART5_IRQn #define UART_DEBUG_IRQHandler UART5_IRQHandler #endif 这对吗
最新发布
11-08
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值