首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 软件管理 > 软件开发 >

宏功用

2012-11-08 
宏作用宏定义作用:?(1)方便程序的修改?上面的#define TRUE 1就是一个实例?(2)提高程序的运行效率?宏定义的

宏作用
宏定义作用:?(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');}?程序的输出会是:?123abc??(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"

热点排行