• C/C++宏替换详解


    1. 基本形式

    #define name replacement_text 
    

    通常情况下,#define 指令占一行,替换文本是 define 指令行尾部的所有剩余部分,但也可以把一个较长的宏定义分成若干行,这时需要在待续的行末尾加上一个反斜杠符 ``。

    宏定义也可以带参数,这样可以对不同的宏调用使用不同的替换文本。例:

    #define max(A, B) ((A) > (B) ? (A) : (B)) 
    

    2. 宏展开中的陷阱

    仔细考虑一下 max 的展开式,其中的表达式会被计算两次,因此如果表达式中包含自增运算符或输入/输出等行为,则会出现不正确的情况,例如上述的宏 max

    max(i++, j++)  // wrong 
    

    另外还需要注意,适当使用圆括号以保证计算次序的正确性,例如:

    #define	square(x)	x * x	// wrong 
    

    当用 square(z+1) 调用该宏定义时会出错。

    3. #undef

    在头文件 <stdio.h> 中,getcharputchar 函数在实际中常常被定义为宏,这样可以避免处理字符时调用函数所需的运行时开销。<ctype.h> 头文件中定义的函数也常常是通过宏实现的。

    可以通过 #define 取消名字的宏定义,这样做可以保证后续的调用是函数调用,而不是宏调用:

    #undef getchar
    
    int getchar(void) { ... } 
    

    4. 宏参数、###

    如果在宏定义的替换文本中,参数名以 # 作为前缀则结果将被扩展为由实际参数替换该参数的带引号的字符串。例如,可以将它与字符串连接运算结合起来编写一个调试打印宏:

    #define	dprint(expr)	printf(#expr " = %gn", expr) 
    

    使用语句

    dprint(x/y); 
    

    调用该宏时,该宏将被扩展为:

    printf("x/y" " = %gn", x/y); 
    

    其中的字符串被拼接起来了,这样,该宏调用的效果等价于

    printf("x/y = %gn", x/y); 
    

    预处理器运算符 ## 为宏扩展提供了一种连接实际参数的手段。如果替换文本中的参数与 ## 相邻,则该参数将被实际参数替代,## 与前后的空白符将被删除,并对替换后的结果重新扫描。例如,下面定义的宏 paste 用于连接两个参数:

    #define paste(front, back)	front ## back 
    

    因此,宏调用 paste(name, 1) 的结果将建立记号 name1


    参考文献:

    1. Brian W. Kernighan, Dennis M. Ritchie.The C Programming Language (Second Edition)[M].机械工业出版社:北京,2004:76-77.
  • 相关阅读:
    Data Guard相关参数学习介绍
    Android打包失败Proguard returned with error code 1. See console
    Extjs4 类的定义和扩展
    c#中常用的异常类型
    cocos2d-x过程动作CCProgressTo示例学习笔记
    欧拉函数
    sae上屏蔽错误显示并查看错误日志
    在wdcp环境下架设VSFTPD虚拟用户只上传功能服务器
    Android开发匹配字符笔记
    setImageResource与setImageBitmap的区别
  • 原文地址:https://www.cnblogs.com/faterazer/p/14002205.html
Copyright © 2020-2023  润新知