宏作用

宏定义作用:
 
(1)方便程序的修改
 
上面的#define TRUE 1就是一个实例
 
(2)提高程序的运行效率
 
宏定义的展开是在程序的预处理阶段完成的,无需运行时分配内存,能够部分实现函数的功能,却没有函数调用的压栈、弹栈开销,效率较高

(3)增强可读性
 
这点不言而喻,当我们看到类似PI这样的宏定义时,自然可以想到它对应的是圆周率常量
 
(4)字符串拼接
 
 例如:
#define CAT(a,b,c) a##b##c
 
main()
{
    printf("%d\n" CAT(1,2,3));
    printf("%s\n", CAT('a', 'b', 'c');
}
 
程序的输出会是:
 
123
abc
 
 
(5)参数转化成字符串
 
示例:
 
#defind CAT(n) "abc"#n
 
main()
{
    printf("%s\n", CAT(15));
}
 
输出的结果会是
abc15
 
(6)用于程序调试跟踪

常见的用于调试的宏有,_ L I N E _,_ F I L E _,_ D A T E _,_ T I M E _,_ S T D C _

(7)实现可变宏

举例来说:

#define PR(...) printf(_ _VA_ARGS_ _)

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 exampleNum(n) num##n

           int num9=9;

             使用:

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

             注意:

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

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

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

    另:如果##后的参数本身也是一个宏的话,##会阻止这个宏的展开

    #define STRCPY(a, b)   strcpy(a ## _p, #b)

            int main()
            {
                char var1_p[20];
                char var2_p[30];
                strcpy(var1_p, "aaaa");
                strcpy(var2_p, "bbbb");
                STRCPY(var1, var2);
                STRCPY(var2, var1);
                printf("var1 = %s\n", var1_p);
                printf("var2 = %s\n", var2_p);
                return 0;
              
               STRCPY(STRCPY(var1,var2),var2);
               //这里是否会展开为: strcpy(strcpy(var1_p,"var2")_p,"var2“)?答案是否定的:
               //展开结果将是:  strcpy(STRCPY(var1,var2)_p,"var2")
               //## 阻止了参数的宏展开!如果宏定义里没有用到 # 和 ##, 宏将会完全展开
            }  

      3、 # @   charizing 字符化操作符

             作用: 将传入单字符参数名转换成字符 以一对单引用括起来

       如:

                 #define makechar(x)  #@x

   a = makechar(b);

    展开后变成了:

   a= 'b';

   4、\ 续行操作符

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

4.例子:
举列 -- 试比较下述几个宏定义的区别

#define A1(name, type) type name_##type##_type 或
#define A2(name, type) type name##_##type##_type

A1(a1, int); /* 等价于: int name_int_type; */
A2(a1, int); /* 等价于: int a1_int_type; */

解释:
1) 在第一个宏定义中,"name"和第一个"_"之间,以及第2个"_"和第二个
"type"之间没有被分隔,所以预处理器会把name_##type##_type解释成3段:
“name_”、“type”、以及“_type”,这中间只有“type”是在宏前面出现过
的,所以它可以被宏替换。

2) 而在第二个宏定义中,“name”和第一个“_”之间也被分隔了,所以
预处理器会把name##_##type##_type解释成4段:“name”、“_”、“type”
以及“_type”,这其间,就有两个可以被宏替换了。

3) A1和A2的定义也可以如下:
#define A1(name, type) type name_ ##type ##_type
<##前面随意加上一些空格>
#define A2(name, type) type name ##_ ##type ##_type

结果是## 会把前面的空格去掉完成强连接,得到和上面结果相同的宏定义

单独的一个 #

至于单独一个#,则表示 对这个变量替换后,再加双引号引起来。比如

#define __stringify_1(x) #x
那么
__stringify_1(linux) <==> "linux"

猜你喜欢

转载自talentluke.iteye.com/blog/1694158