define宏定义中的#,##,@#及\符号

define宏定义中的#,##,@#及\符号

在#define中,标准只定义了#和##两种操作。#用来把参数转换成字符串,##则用来连接两个前后两个参数,把它们变成一个字符串。

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 sum(a,b) (a+b);中a和b均为某一参数的代表符号,即形式参数。

而##的作用则是将宏定义的多个形参成一个实际参数名。

如:

#define exampleNum(n) num##n

int num9=9;

使用:

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

注意:

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

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

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

// preprocessor_token_pasting.cpp

#include <stdio.h>

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

int token9 = 9;

int main()

{

paster(9);

}

运行结果:

token9 = 9

3、@# (charizing)字符化操作符。

只能用于有传入参数的宏定义中,且必须置于宏定义体中的参数名前。作用是将传的单字符参数名转换成字符,以一对单引用括起来。

#define makechar(x)  #@x

a = makechar(b);

展开后变成了:

a= ‘b‘;

4、\ 行继续操作符

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

注意:换行不能切断单词,只能在空格的地方进行。

宏定义的特殊符号# ## - [C++]

1. 利用宏参数创建字符串:# 运算符
 
    在类函数宏(function-like macro)的替换部分中,“#”符号用作一个预处理运算符,它可以把语言符号(token)转化为字符串。例如,如果 x 是一个宏参量,那么 #x 可以把参数名转化为相应的字符串。该过程称为字符串化。
 
    说明:类函数宏就是带参数的宏。类函数宏的定义中,用圆括号括起来一个或多个参数,随后这些参数出现在替换部分。
#include <stdio.h>    
#define PSQR(x) printf("The square of " #x " is %d. \r\n", (x) * (x))    
    
int main(void)   
{   
    int y = 5;    
       
    PSQR(y);   
    PSQR(2 + 4);    
    
    return 0;    
}   
    
// 输出:   
The square of y is 25.     // 用 "y" 代替 #x    
The square of 2 + 4 is 36. // 用 "2 + 4" 代替 #x  
#include <stdio.h> 
#define PSQR(x) printf("The square of " #x " is %d. \r\n", (x) * (x)) 
 
int main(void)
{
    int y = 5; 
    
    PSQR(y);
    PSQR(2 + 4); 
 
    return 0; 
}
 
// 输出:
The square of y is 25.     // 用 "y" 代替 #x 
The square of 2 + 4 is 36. // 用 "2 + 4" 代替 #x
#define STRING2(x) #x   
#define STRING(x) STRING2(x)   
    
#define WQ wangqi    
    
#pragma message(STRING2(WQ)) // WQ(字符串)   
#pragma message(STRING(WQ))  // wangqi(字符串)  
#define STRING2(x) #x
#define STRING(x) STRING2(x)
 
#define WQ wangqi 
 
#pragma message(STRING2(WQ)) // WQ(字符串)
#pragma message(STRING(WQ))  // wangqi(字符串) 
2. 预处理器的粘合剂:## 运算符
 
    和 # 运算符一样,## 运算符可以用于类函数宏的替换部分。另外,## 运算符还可用于类对象宏(object-like macro)的替换部分。这个运算符把两个语言符号组合成单个语言符号。例如,可以定义如下宏:
#define XNAME(n) x ## n  
#define XNAME(n) x ## n
    宏调用 XNAME(4) 会展开成 x4 。
 
    说明:类对象宏就是用来代表值的宏。如,#define PI 3.141593 中的PI。
#include <stdio.h>    
#define XNAME(n) x ## n    
#define PRINT_XN(n) printf("x" #n " = %d\r\n", x ## n);    
    
int main(void)   
{   
    int XNAME(1) = 14; // 变为 int x1 = 14;    
    int XNAME(2) = 20; // 变为 int x2 = 20;    
    PRINT_XN(1)        // 变为 printf("x1 = %d\r\n", x1);    
    PRINT_XN(2)        // 变为 printf("x2 = %d\r\n", x2);    
    
    return 0;    
}   
    
// 输出:   
x1 = 14    
x2 = 20   
#include <stdio.h> 
#define XNAME(n) x ## n 
#define PRINT_XN(n) printf("x" #n " = %d\r\n", x ## n); 
 
int main(void)
{
    int XNAME(1) = 14; // 变为 int x1 = 14; 
    int XNAME(2) = 20; // 变为 int x2 = 20; 
    PRINT_XN(1)        // 变为 printf("x1 = %d\r\n", x1); 
    PRINT_XN(2)        // 变为 printf("x2 = %d\r\n", x2); 
 
    return 0; 
}
 
// 输出:
x1 = 14 
x2 = 20 
#define __T(x)      L ## x   
#define _T(x)       __T(x)   
#define _TEXT(x)    __T(x)   
        
#define WQ "wangqi"    
        
#pragma message(__T(WQ)) // LWQ (标识符)   
wcout << _T(WQ);         // wangqi(宽字节字符串)  
#define __T(x)      L ## x
#define _T(x)       __T(x)
#define _TEXT(x)    __T(x)
     
#define WQ "wangqi" 
     
#pragma message(__T(WQ)) // LWQ (标识符)
wcout << _T(WQ);         // wangqi(宽字节字符串) 
3. 语言符号 
 
    从技术方面看,系统把宏的主体当作语言符号(token)类型字符串,而不是字符型字符串。C 预处理器中的语言符号是宏定义主体中的单独的“词(word)”。用空白字符把这些词分开。例如:
#define FOUR 2*2  
#define FOUR 2*2
    这个定义中有一个语言符号:即序列 2*2 。但是:
#define SIX 2 * 3  
#define SIX 2 * 3
    这个定义中有三个语言符号:2、* 和 3 。
 
    在处理主体中的多个空格时,字符型字符串和语言符号型字符串采用不同方法。考虑下面的定义:
#define EIGHT 4    *    8  
#define EIGHT 4    *    8
    把主体解释为字符型字符串时,预处理器用 4    *    8 替换 EIGHT 。也就是说,额外的空格也当作替换文本的一部分。但是,当把主体解释为语言符号类型时,预处理器用由单个空格分隔的三个语言符号,即 4 * 8 来替换 EIGHT 。 换句话说,用字符型字符串的观点看,空格也是主体的一部分;而用语言符号字符串的观点看,空格只是分隔主体中语言符号的符号。在实际应用中,有些 C 编译器把宏主体当作字符串而非语言符号。在比这个实例更复杂的情况下,字符与语言符号之间的差异才有实际意义。
 
    顺便提一下,C 编译器处理语言符号的方式比预处理器的处理方式更加复杂。编译器能理解 C 的规则,不需要用空格来分隔语言符号。例如,C 编译器把 2*2 当作三个语言符号。原因是 C 编译器认为每个 2 都是一个常量,而 * 是一个运算符。

define宏定义中的#,##,@#及\符号,码迷,mamicode.com

时间: 2024-10-11 07:13:55

define宏定义中的#,##,@#及\符号的相关文章

const基础知识、const在C++和C中的异同、const与#define宏定义的异同

1.const基础知识 (1)用const定义常量 const int a = 5;   //定义了一个int型常量a,其值为5 (注:在C++中将其存储在符号表中(key,value),并不分配内存空间,只有在取地址或者定义为全局变量在其他文件中使用时才分配内存空间) (2)const与指针 1 const int* p1 = &a; //p1是指向常量a的指针,a的值不能通过指针p1改变 2 3 int* const p2 =&b; //p2是只能指向b的指针,b的值可以通过p2改变

宏定义中的#,##,...,do{}while(0),__VA_ARGS__

宏定义中的#,## 1.在一个预处理器宏中的参数前面使用一个#,预处理器会把这个参数转换为一个字符数组 #define syslog(a) fprintf(stderr,"Warning: " #a"\n"); 2.简单的说,"## "是一种分隔连接方式,它的作用是先分隔,然后进行强制连接 举列 -- 试比较下述几个宏定义的区别 #define A1(name, type)  type name_##type##_type 或 #define A

typedef 类型重命名 和 #define 宏定义(1)

http://www.blogjava.net/jasmine214--love/archive/2010/11/29/339307.html 在现实生活中,信息的概念可能是长度,数量和面积等.在C语言中,信息被抽象为int.float和double等基本数据类型.从基本数据类型名称上, 不能够看出其所代表的物理属性,并且int.float和double为系统关键字,不可以修改.为了解决用户自定义数据类型名称的需求,C语言中引入类 型重定义语句typedef,可以为数据类型定义新的类型名称,从而

宏定义中##和#的作用

解释1 : 定义中##和#的作用内核中有很多的宏定义,在宏定义define中经常看到两个字符串##和#,这里把它的用法做一下说明:1. ####是一个连接符号,用于把参数连在一起例如:> #define FOO(arg) my##arg则> FOO(abc)相当于 myabc2.#:是"字符串化"的意思.出现在宏定义中的#是把跟在后面的参数转换成一个字符串例如:> #define STRCPY(dst, src) strcpy(dst, #src)则> STRC

C宏定义中的 #,##,#@

宏定义中会出现#xxx ,A###B,以及微软独自特有的#@xxx.分别解释. 1 define SB(x) #x 它的作用是把输入的东西转换为字符串 string str = SB(123); 2 define BB(x) UXX##x 这个是个符号连接的作用,把UXX(随意的跟符号x连接成一个新的符号,这里说的符号就是变量的意思 int BB(1); U1 = 100; 3 define CB(x) #@x 作用会是把x的最后一个字符转换成字符,这个是Windows独有的. char b =

#和##在宏定义中使用问题

转自:http://www.linuxidc.com/Linux/2014-06/102925.htm 有一道经典的C语言问题,关于宏定义中#和##符号的使用和宏定义展开问题 程序如下: #include <stdio.h>#define f(a,b) a##b#define g(a)  #a#define h(a) g(a) int main(){        printf("%s\n", h(f(1,2)));        printf("%s\n&quo

iOS define 宏定义 和 const定义常量区别

const   const 是c++中的修饰符.  c++中常用来定义常量,修饰左值. #define 宏定义语句, 在预处理阶段直接做文本替换,不做类型检查. 它们之间的最大区别: 1.  对于const 变量,系统只给了一个相应的内存地址,而#define则是给出了一个立即数.因为const变量是存放在内存的静态区域中,所以在程序运行过程中const变量只有一个拷贝,而#define 所定义的宏变量却有多个拷贝,所以宏在程序运行过程中所消耗的内存要比const变量的大得多.#define所定

iOS 7:漫谈#define 宏定义(转)

iOS 7:漫谈#define 宏定义 #define宏定义在C系开发中可以说占有举足轻重的作用.底层框架自不必说,为了编译优化和方便,以及跨平台能力,宏被大量使用,可以说底层开发离开define将寸步难行.而在更高层级进行开发时,我们会将更多的重心放在业务逻辑上,似乎对宏的使用和依赖并不多.但是使用宏定义的好处是不言自明的,在节省工作量的同时,代码可读性大大增加. 如果想成为一个能写出漂亮优雅代码的开发者,宏定义绝对是必不可少的技能(虽然宏本身可能并不漂亮优雅XD).但是因为宏定义对于很多人来

iOS 7:漫谈#define 宏定义

#define宏定义在C系开发中可以说占有举足轻重的作用.底层框架自不必说,为了编译优化和方便,以及跨平台能力,宏被大量使用,可以说底层开发离开define将寸步难行.而在更高层级进行开发时,我们会将更多的重心放在业务逻辑上,似乎对宏的使用和依赖并不多.但是使用宏定义的好处是不言自明的,在节省工作量的同时,代码可读性大大增加. 如果想成为一个能写出漂亮优雅代码的开发者,宏定义绝对是必不可少的技能(虽然宏本身可能并不漂亮优雅XD).但是因为宏定义对于很多人来说,并不像业务逻辑那样是每天会接触的东西