当前位置:编程学习 > C/C++ >>

C语言预处理功能——关于字符串化和符号粘贴

 

在C语言开发中,宏定义是一个非常有用的工具,它可以使我们的代码更容易理解,更容易维护。如查一个常量在多处被使用,且今后可能会根据不同需要而修改的话,将其define一下那是再好不过了。除此之外,宏定义还有其他的功能,了解它,将更好地辅助我们开发C程序。先看个例子:

 

#define Conn(x,y)   x##y 

#define ToString(x) #x 

#define ToChar(x)   #@x

 

这几个宏定义中分别用到了“##”、“#”,“#@”它们的功能分别是:

1、粘接操作符##——连接两个宏名,注意所连接的是宏名,而不是其所指代的值;

如int Conn(a,b);定义了一个int型变量ab,以后可以直接调用ab,而不必采用Conn(a,b)的形式;

printf(Conn("ab","cd"));输出结果为:abcd

但是:

 

#define M 0 

#define var(x) Var_##x 

... 

 

int var(M);//此处定义了什么?

 

根据ANSI/ISO C,##操作符只是简单地粘接两个宏名,则int var(M)定义的应该就是Var_M,据测试,这一点是没有问题的。但在一些比较古老的C编译环境中,也有可能定义的是Var_0,比如在TC 2.0中测试发现“Var_0=0”可编译通过,而“Var_M=0”出现ERROR。

2、字符串化操作符#——将宏名转化为字符串

如printf(ToString(var1));输出结果为:var1。var1可以是一个已定义的变量名,也可以是一个从未出现的字符组合

类似地,若有:

 

#define STR 0  

... 

 

printf(Tostring(STR));// 

 

当前流行编译环境会输出STR,而TC 2.0则会输出0。

3、字符化操作#@——将宏名转化为字符,注意:早期编译器可能不支持

如:

 

char c; 

 

c = ToChar(1);//c = '1' 

c = ToChar(a);//c = 'a'

 

如果提供的宏中不止一个字符(注意,不能超过4个,否则编译器报错),则转化结果为最后一个字符,如

 

c = ToChar(abc);//c = 'c' 

c = ToChar(abcd);//c= 'd' 

c = ToChar(abcde);//ERROR

总结一下,关于其用法是自己总结的,肯定不全。

 

1、使用中遵循ANSI C中规定,但要记得编译通不过是可能是早期编译器不支持C标准的问题;

2、##操作可应用在变量定义中,若程序开发中遇到要定义一大堆变量,且这些变量具有相同的前缀时,##很显得尤为重要,它可以使代码更加整洁,且减少了出错的机率。如果一旦后来发现需要批量修改变量前缀,你会庆幸自己使用了这么一件利器;

3、#操作符可用于调试时将变量名输出,可配合##一起使用,如定义#define CHECK_VAR(x,fmt) printf(#x " = " #fmt "\n", x),则CHECK_VAR(var1,%d)相当于printf("var = %d\n", var1);

Tips:

1、ANSI C中规定若宏定义名出现在引号(' '或" ")中,则不进行替换,但有些早期编译器的处理可能有所不同,如#define CTRL(c) (‘c’& 37),按照标准CTRL(d)被扩展成('c' & 37)。显然,这没有完成作者的本意,它在某些编译器下碰巧能工作不过是个意外,实际使用中应避免。

2、字符串的连接不必使用##这么麻烦,实际中两个字符串常量可以直接写到一起,如printf("ab""cd")输出abcd。或在使用##宏定义时,可以用printf(ToString(str) "\n");输出字符串后换行,以前不敢这样用,后来试验了下发现还比较好使,当然直接用puts也可完成同样功能。

3、若要使#或##转换的是宏字符常量的值也不是其名字,可以使用间接方法,如:

#define ToString(x) #x 

#define Xstr(x) ToString(x) 

#define STR1 STR2 

... 

 

printf(Xstr(STR1));//输出结果是STR2而非STR1

 

本文出自 “学须静也” 博客

补充:软件开发 , C语言 ,
CopyRight © 2022 站长资源库 编程知识问答 zzzyk.com All Rights Reserved
部分文章来自网络,