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


    define宏定义中的###@#符号

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

    1stringizing)字符串化操作符。其作用是:将宏定义中的传入参数名转换成用一对双引号括起来参数名字符串。其只能用于有传入参数的宏定义中,且必须置于宏定义体中的参数名前。

    如:

    #define example(instr) printf("the input string is: %s ",#instr)

    #define example1(instr) #instr

    当使用该宏定义时:

    example(abc); 在编译时将会展开成:printf("the input string is: %s ","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);中ab均为某一参数的代表符号,即形式参数。

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

    如:

    #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 可以把参数名转化为相应的字符串。该过程称为字符串化。
     
        说明:类函数宏就是带参数的宏。类函数宏的定义中,用圆括号括起来一个或多个参数,随后这些参数出现在替换部分。
    #include <stdio.h>    
    #define PSQR(x) printf("The square of " #x " is %d.  ", (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.  ", (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 ", 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 ", x1);    
        PRINT_XN(2)        // 变为 printf("x2 = %d ", x2);    
        
        return 0;    
    }   
        
    // 输出:   
    x1 = 14    
    x2 = 20   
    #include <stdio.h> 
    #define XNAME(n) x ## n 
    #define PRINT_XN(n) printf("x" #n " = %d ", 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 ", x1); 
        PRINT_XN(2)        // 变为 printf("x2 = %d ", 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)类型字符串,而不是字符型字符串。预处理器中的语言符号是宏定义主体中的单独的词(word。用空白字符把这些词分开。例如:
    #define FOUR 2*2  
    #define FOUR 2*2
        这个定义中有一个语言符号:即序列 2*2 。但是:
    #define SIX 2 * 3  
    #define SIX 2 * 3
        这个定义中有三个语言符号:2和 
     
        在处理主体中的多个空格时,字符型字符串和语言符号型字符串采用不同方法。考虑下面的定义:
    #define EIGHT 4    *    8  
    #define EIGHT 4    *    8
        把主体解释为字符型字符串时,预处理器用 4    *    8 替换 EIGHT 。也就是说,额外的空格也当作替换文本的一部分。但是,当把主体解释为语言符号类型时,预处理器用由单个空格分隔的三个语言符号,即 4 * 8 来替换 EIGHT 。 换句话说,用字符型字符串的观点看,空格也是主体的一部分;而用语言符号字符串的观点看,空格只是分隔主体中语言符号的符号。在实际应用中,有些 编译器把宏主体当作字符串而非语言符号。在比这个实例更复杂的情况下,字符与语言符号之间的差异才有实际意义。
     
        顺便提一下,编译器处理语言符号的方式比预处理器的处理方式更加复杂。编译器能理解 的规则,不需要用空格来分隔语言符号。例如,编译器把 2*2 当作三个语言符号。原因是 编译器认为每个 都是一个常量,而 是一个运算符。
     

  • 相关阅读:
    这2天参加WinHEC大会,园子里以有很多介绍,就不多说了,会上用手机录了一段windows 最新触摸屏操作技术演示,可以看看
    自启动U盘,一个会流行的好玩意
    为什么数据库导入是自动增量属性自动消失乐呢?
    网络带宽利用率的一般计算方法
    防止ARP病毒的一个小窍门
    Windows 系统补丁管理策略
    PDC大会就要召开了,园里有去的吗,看到一片文章,不知道windows7是否真的很好
    门户框架在项目和产品中的使用心得
    这段时间开发了一个共享软件,主要做IP资源管理的(SmartIPView),大家有兴趣可以看看,或给指点指点
    OpenGL自学教程1(窗口建立)
  • 原文地址:https://www.cnblogs.com/TTTTT/p/3701871.html
Copyright © 2020-2023  润新知