c语言宏字符连接嵌套,C语言宏定义##链接符和#符的使用

#include数组

#define f(a,b) a##bide

#define g(a) #a函数

#define h(a) g(a)ui

int main()lua

{spa

printf("%s\n",h(f(1,2)));.net

printf("%s\n",g(f(1,2)));翻译

return 0;指针

}

在宏定义里,a##b就是把a,b联接起来,

好比f(1,2)就是12,可是是数。

#a就是把a转化成字串,并合并。

因此 printf("%s\n",g(f(1,2)));就直接把f(1,2)转成字串了。

#define A(x) T_##x

#define B(x) #@x

#define C(x) #x

咱们假设:x=1,则有:

A(1)------〉T_1

B(1)------〉'1'

C(1)------〉"1"

C语言中如何使用宏C(和C++)中的宏(Macro)属于编译器预处理的范畴,属于编译期概念(而非运行期概念)。下面对常遇到的宏的使用问题作了简单总结。

关于#和##

在C语言的宏中,#的功能是将其后面的宏参数进行字符串化操做(Stringfication),简单说就是在对它所引用的宏变量 经过替换后在其左右各加上一个双引号。好比下面代码中的宏:

#define WARN_IF(EXP) do{ if (EXP) fprintf(stderr, "Warning: " #EXP "/n"); } while(0)

那么实际使用中会出现下面所示的替换过程:

WARN_IF (divider == 0);

被替换为

do {

if (divider == 0)

fprintf(stderr, "Warning" "divider == 0" "/n");

} while(0);

这样每次divider(除数)为0的时候便会在标准错误流上输出一个提示信息。

而##被称为链接符(concatenator),用来将两个Token链接为一个Token。注意这里链接的对象是Token就行,而不必定 是宏的变量。好比你要作一个菜单项命令名和函数指针组成的结构体的数组,而且但愿在函数名和菜单项命令名之间有直观的、名字上的关系。那么下面的代码就很是实用:

struct command

{

char * name;

void (*function) (void);

};

#define COMMAND(NAME) { NAME, NAME ## _command }

// 而后你就用一些预先定义好的命令来方便的初始化一个command结构的数组了:

struct command commands[] = {

COMMAND(quit),

COMMAND(help),

...

}

COMMAND宏在这里充当一个代码生成器的做用,这样能够在必定程度上减小代码密度,间接地也能够减小不留心所形成的错误。咱们还能够n个##符号链接 n+1个Token,这个特性也是#符号所不具有的。好比:

#define LINK_MULTIPLE(a,b,c,d) a##_##b##_##c##_##d

typedef struct _record_type LINK_MULTIPLE(name,company,position,salary);

// 这里这个语句将展开为:

// typedef struct _record_type name_company_position_salary;

## 链接符号由两个井号组成,其功能是在带参数的宏定义中将两个子串(token)联接起来,从而造成一个新的子串。但它不能够是第一个或者最后一个子串。所谓的子串(token)就是指编译器可以识别的最小语法单元。

#符是把传递过来的参数当成字符串进行替代。

下面来看看它们是怎样工做的。这是MSDN上的一个例子。 假设程序中已经定义了这样一个带参数的宏:

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

同时又定义了一个整形变量: int token9 = 9;

如今在主程序中如下面的方式调用这个宏: paster( 9 );

那么在编译时,上面的这句话被扩展为: printf( "token" "9" " = %d", token9 );

注意到在这个例子中,paster(9);中的这个”9”被原封不动的当成了一个字符串,与”token”链接在了一块儿,从而成为了token9。而#n也被”9”所替代。 可想而知,上面程序运行的结果就是在屏幕上打印出token9=9

定义单行宏:主要有如下三种用法.

1) 前加##或后加##,将标记做为一个合法的标识符的一部分.注意,不是字符串.多用于多行的宏定义中.例如:

#define A(x)  T_##x

则 int A(1) = 10; //等效于int T_1 = 10;

#define A(x)  Tx##__

则 int A(1) = 10; //等效于int T1__ = 10;

2) 前加#@,将标记转换为相应的字符,注意:仅对单一标记转换有效

#define B(x) #@x

则B(a)即’a’,B(1)即’1’.但B(abc)却不甚有效.

3) 前加#,将标记转换为字符串.

#define C(x) #x

则C(1+1) 即 ”1+1”.

关于...的使用

...在C宏中称为Variadic Macro,也就是变参宏。好比:

#define myprintf(templt,...) fprintf(stderr,templt,__VA_ARGS__)

// 或者

#define myprintf(templt,args...) fprintf(stderr,templt,args)

第一个宏中因为没有对变参起名,咱们用默认的宏__VA_ARGS__来替代它。第二个宏 中,咱们显式地命名变参为args,那么咱们在宏定义中就能够用args来代指变参了。同C语言的stdcall同样,变参必须做为参数表的最有一项出 现。当上面的宏中咱们只能提供第一个参数templt时,C标准要求咱们必须写成:

myprintf(templt,);

的形式。这时的替换过程为:

myprintf("Error!/n",);

替换为:

fprintf(stderr,"Error!/n",);

这是一个语法错误,不能正常编译。这个问题通常有两个解决方法。首先,GNU CPP提供的解决方法容许上面的宏调用写成:

myprintf(templt);

而它将会被经过替换变成:

fprintf(stderr,"Error!/n",);

很明显,这里仍然会产生编译错误(非本例的某些状况下不会产生编译错误)。除了这种方式外,c99和GNU CPP都支持下面的宏定义方式:

#define myprintf(templt, ...) fprintf(stderr,templt, ##__VAR_ARGS__)

这时,##这个链接符号充当的做用就是当__VAR_ARGS__为空的时候,消除前面的那个逗号。那么此时的翻译过程以下:

myprintf(templt);

被转化为:

fprintf(stderr,templt);

这样若是templt合法,将不会产生编译错误。 这里列出了一些宏使用中容易出错的地方,以及合适的使用方式。

错误的嵌套-Misnesting

宏的定义不必定要有完整的、配对的括号,可是为了不出错而且提升可读性,最好避免这样使用。

由操做符优先级引发的问题-Operator Precedence Problem

因为宏只是简单的替换,宏的参数若是是复合结构,那么经过替换以后可能因为各个参数之间的操做符优先级高于单个参数内部各部分之间相互做用的操做符优先级,若是咱们不用括号保护各个宏参数,可能会产生预想不到的情形。好比:

#define ceil_div(x, y) (x + y - 1) / y

那么

a = ceil_div( b & c, sizeof(int) );

将被转化为:

a = ( b & c + sizeof(int) - 1) / sizeof(int);

// 因为+/-的优先级高于&的优先级,那么上面式子等同于:

a = ( b & (c + sizeof(int) - 1)) / sizeof(int);

这显然不是调用者的初衷。为了不这种状况发生,应当多写几个括号:

#define ceil_div(x, y) (((x) + (y) - 1) / (y))

消除多余的分号-Semicolon Swallowing

一般状况下,为了使函数模样的宏在表面上看起来像一个一般的C语言调用同样,一般状况下咱们在宏的后面加上一个分号,好比下面的带参宏:

MY_MACRO(x);

可是若是是下面的状况:

#define MY_MACRO(x) {/* line 1 *//* line 2 *//* line 3 */ }

//...

if (condition())

MY_MACRO(a);

else

{...}

这样会因为多出的那个分号产生编译错误。为了不这种状况出现同时保持MY_MACRO(x);的这种写法,咱们须要把宏定义为这种形式:

#define MY_MACRO(x) do {

/* line 1 *//* line 2 *//* line 3 */ } while(0)

这样只要保证老是使用分号,就不会有任何问题。

Duplication of Side Effects

这里的Side Effect是指宏在展开的时候对其参数可能进行屡次Evaluation(也就是取值),可是若是这个宏参数是一个函数,那么就有可能被调用屡次从而达到不一致的结果,甚至会发生更严重的错误。好比:

#define min(X,Y) ((X) > (Y) ? (Y) : (X))

//...

c = min(a,foo(b));

这时foo()函数就被调用了两次。为了解决这个潜在的问题,咱们应当这样写min(X,Y)这个宏:

#define min(X,Y) ({typeof (X) x_ = (X);typeof (Y) y_ = (Y);(x_ < y_) ? x_ : y_; })

({...})的做用是将内部的几条语句中最后一条的值返回,它也容许在内部声明变量(由于它经过大括号组成了一个局部Scope)。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值