C语言中#的神奇作用

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/jackailson/article/details/50989106

      By:Ailson Jack

      Date:2016.03.26

      个人博客:www.only2fire.com

      本文在我博客的地址是:http://www.only2fire.com/archives/56.html,排版更好,便于学习。

      C语言中,在宏里面的’#’和’##’有它非常神奇的作用,下面就来说说具体的用法。

1、一般用法

      我们使用#把宏参数变为一个字符串,用##把两个宏参数贴合在一起。

      下面的代码是演示代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14

#include <stdio.h>

#define STRING(s)       #s
#define CONNECT(a,b)    int(a##e##b)

int main(void)
{
    //输出字符串"abcdefg"
    printf("string:%s\r\n",STRING(abcdefg));
    //2e3输出:2000
    printf("connect:%d\r\n",CONNECT(2,3));

    return 0;
}

      运行结果如下图:

扫描二维码关注公众号,回复: 3037651 查看本文章

2、当宏参数是另一个宏的时候

       需要注意的是凡是宏定义里有用’#’或’##’的地方宏参数是不会再展开。

1)、非’#’和’##’的情况 

#define TOW              (2)

#define MUL(a,b)        (a*b)

printf(“%d*%d=%d\n”, TOW, TOW, MUL(TOW,TOW));

      这行的宏会被展开为:

printf(“%d*%d=%d\n”, (2), (2), ((2)*(2)));

      MUL里的参数TOW会被展开为(2)。

2)、当有’#’或’##’的时候 

#define A                    (2)

#define STR(s)           #s

#define CONS(a,b)    int(a##e##b)

printf(“int max: %s\n”, STR(INT_MAX)); //INT_MAX 这行会被展开为:printf(“int max: %s\n”, “INT_MAX”);

printf(“%s\n”, CONS(A, A));              //这一行则是:printf(“%s\n”, int(AeA));

      INT_MAX和A都不会再被展开,然而解决这个问题的方法很简单。加多一层中间转换宏。加这层宏的用意是把所有宏的参数在这层里全部展开,那么在转换宏里的那一个宏(_STR)就能得到正确的宏参数。

#define A                       (2)

#define _STR(s)           #s

#define STR(s)              _STR(s)      //转换宏

#define _CONS(a,b)     int(a##e##b)

#define CONS(a,b)       _CONS(a,b)         //转换宏

printf(“int max: %s\n”, STR(INT_MAX)); // INT_MAX,int型的最大值,为一个变量

      输出为: int max: 0x7fffffff,STR(INT_MAX) –> _STR(0x7fffffff) 然后再转换成字符串。

printf(“%d\n”, CONS(A, A));

      输出为:200,CONS(A, A) –> _CONS((2), (2)) –> int((2)e(2)) 。

3、’#’和’##’的一些应用特例

1)、合并匿名变量名

#define   ___ANONYMOUS1(type,var,line)    type   var##line

#define   __ANONYMOUS0(type, line)           ___ANONYMOUS1(type, _anonymous, line)

#define   ANONYMOUS(type)                         __ANONYMOUS0(type, __LINE__)

例:ANONYMOUS(static int);即: static int _anonymous70;//70表示该行行号;

第一层:ANONYMOUS(static int);   –>   __ANONYMOUS0(static int, __LINE__);

第二 层:__ANONYMOUS0(static int, __LINE__);   –>   ___ANONYMOUS1(static int, _anonymous, 70);

第三层:___ANONYMOUS1(static int, _anonymous, 70);    –>   static int   _anonymous70;

即每次只能解开当前层的宏,所以__LINE__在第二层才能被解开

2)、记录文件名

#define   _GET_FILE_NAME(f)    #f

#define   GET_FILE_NAME(f)      _GET_FILE_NAME(f)

static char  FILE_NAME[] = GET_FILE_NAME(__FILE__); 

其中2用得比较多,很方便。

      注:转载请注明出处,谢谢!^_^

猜你喜欢

转载自blog.csdn.net/jackailson/article/details/50989106
今日推荐