C/C++ 中的宏/Macro
|
宏(Macro)本质上就是代码片段,通过别名来使用。在编译前的预处理中,宏会被替换为真实所指代的代码片段,即下图中 Preprocessor 处理的部分。
C/C++ 代码编译过程 - 图片来自 ntu.edu.sg 根据用法的不同,分两种,Object-like 和 Function-like。前者用于 Object 对象,后者用于函数方法。 C/C++ 代码编译过程中,可通过相应参数来获取到各编译步骤中的产出,比如想看被预处理编译之后的宏,使用 $ gcc -E macro.c 宏的定义通过 #define NAME_OF_MACRO value 比如,以下代码定义了一个名为 #define BUFFER_SIZE 1024 使用时, foo = (char *) malloc (BUFFER_SIZE); 使用预处理器编译: $ gcc -E test.c 编译结果: foo = (char *) malloc (1024); 多行宏的定义是跟随 #include <stdio.h> 多行的宏经过编译后会还原到一行中。 test.c #include <stdio.h> 编译后: int main() {
宏展开时的顺序宏的展开是在处理源码时按照其出现位置进行的,如果宏定义有嵌套关系,也是层层进行展开,比如: #include <stdio.h> 首先遇到 int main() {
其展开的顺序并不是宏定义时的顺序,为了验证,可将上面示例代码中两个宏的定义调换一下,得到: -#define GREETING_NAME "wayou" 再次编译查看产出,会发现没有区别,也不会报 像下面这样,当宏存在覆盖时,会以新的为准,其结果为 37。 #define BUFSIZE 1020 Object-like 宏Object-like 类型的宏看起来就像普通的数据对象,故名。多用于数字常量的情形下。且宏名一般使用全大写形式方便识别。像上面示例中,都是 Object-like 的。 Function-like 宏也可定义出使用时像是方法调用一样的宏,这便是 Function-like 类型的宏。 #define lang_init() c_init() 函数类型的宏只在以方法调用形式使用时才会被展开,即名称后加括号,否则会被忽略。当宏名和函数名重名时,这一策略就会显得有用了,比如: extern void foo(void); 这里 函数类型的宏在定义时需注意,宏名与后面括号不能有空格,否则就是普通的 Object-like 类型对象。 #define lang_init () c_init() 宏的参数函数类型的宏,可以像正常函数一样指定入参,入参需为逗号分隔合法的 C 字面量。 #define min(X, Y) ((X) < (Y) ? (X) : (Y)) 入参中的括号入参中只需要括号对称,但不要求方括号或花括号成对出现,所以下面的代码: macro (array[x = y, x + 1]) 其入参实际为 入参的展开入参本质上也是宏,对象类型的宏,在函数宏展示时,这些参数也被展示到了函数宏的函数体里。 min (min (a, b), c) 首先被展开成: min (((a) < (b) ? (a) : (b)), (c)) 然后进一步展开成(此处换行为方便阅读,实际编译后没有): ((((a) < (b) ? (a) : (b))) < (c) 参数的缺省函数宏在使用时其入参可缺省,但不能全部缺省,至少提供一个入参。 min(, b) → (( ) < (b) ? ( ) : (b)) 字符化/Stringizing如果函数宏中入参在字符串中,是不会被展开的,它就是普通的字符串字面量,这样的结果是符合预期的。 #define foo(x) x, "x" 但如果确实想将入参展开成字符串,可在使用入参时,加上 #define WARN_IF(EXP) \ 此处 拼接通过 宏拼接一般用在需要拼接的宏是来自宏参数的情况,其他情况,大可直接将两个宏写在一起即可,用不着 考察下面这个场景,其中命令名重复出现: struct command 通过定义宏配合拼接,可达到精简代码的目的: #define COMMAND(NAME) { #NAME, NAME ## _command }
struct command commands[] =
不定参数像普通函数一样,函数类型的宏也可定义接收不定参数。 #define eprintf(…) fprintf (stderr, __VA_ARGS__) 调用时,命名参数后面,包括逗号都会进入到 eprintf ("%s:%d: ", input_file, lineno)
// 编译后:
C++ 中可这么写: #define eprintf(args…) fprintf (stderr, args) 不定参数与命名参数混合的情况不定参数为命名参数后面省略的部分。 #define eprintf(format, …) fprintf (stderr, format, __VA_ARGS__) 预设的宏标准库及编译器中预设了一些有用的宏,可以在这里 查阅。 取消和重置宏当某个宏不再使用时,可通过 #define FOO 4 两个宏相似的定义满足以下条件时,我们认为两者是相似的:
比如,下面这些是相似的: #define FOUR (2 + 2) 而下面这些则不然: #define FOUR (2 + 2) 宏重复定义时的表现对于使用了 而如果说一个已经存在的宏,并没有注销,重复定义时,如果相似,则新的定义会忽略,如果不相似,编译器会报警告同时使用新定义的宏。这允许在多个文件中定义同一个宏。 相关资源 |
C/C++ 中的宏/Macro的更多相关文章
- uboot中的中断macro宏
目录 uboot中的中断macro宏 引入 内存分配 流程概览 普通中断 保存现场 中断函数打印具体寄存器 恢复现场 软中断 空间获取 保存现场 附录速记 疑惑待解 title: uboot中的中断m ...
- C中的宏
1. 简单宏定义 简单的宏定义有如下格式: [#define指令(简单的宏)] #define 标识符替换列表 替换列表是一系列的C语言记号,包括标识符.关键字.数.字符常量.字符串字面量.运算符和 ...
- flask第30篇——宏macro和import标签
宏是Jinja2特有的,像Django则没有这个. 先新建一个项目macroDemo: 然后在templates文件夹中新建index.html文件,并在代码中返回渲染后的文件: 然后回到index. ...
- flask中的宏
对于flask中的宏编程.我们使用 macro 来对宏起个名称 宏编程 对于我们来说是减少了代码的重用.以及简化了标签的操作,对与开发效率有很大的提升, 在html中.相信大多数都用到了.input ...
- zabbix上的宏(macro)介绍
宏:macro,预设的文本替换模式: 宏是一种抽象概念(Abstraction),它根据一些列预定义的规则替换一定的文本模式,而解释或编译器在遇到宏时会自动进行这一模式替换.类似地,zabbix基于宏 ...
- Makefile中用宏定义进行条件编译(gcc -D)/在Makefile中进行宏定义-D【转】
本文转载自:http://blog.csdn.net/maopig/article/details/7230311 在源代码里面如果这样是定义的:#ifdef MACRONAME//可选代码#en ...
- Flask基础(15)-->模板代码的复用【宏(Macro)、继承(Block)、包含(include)】
宏 对宏(macro)的理解: 把它看作 Jinja2 中的一个函数,它会返回一个模板或者 HTML 字符串 为了避免反复地编写同样的模板代码,出现代码冗余,可以把他们写成函数以进行重用 需要在多处重 ...
- js中的宏任务与微任务
如果你已经知道了js中存在宏任务和微任务,那么你一定已经了解过promise了.因为在js中promise是微任务的一个入口. 先来看一道题: setTimeout(function(){ conso ...
- C语言的宏macro的使用
C's Macro Introduction 1.The Connect Macros: ## 这是一个预处理连接符,这个操作符主要用来将两个符号连接成为一个完整的宏符号.通过下面的代码,可以看到其具 ...
随机推荐
- django基础之day05,F与Q查询,Q查询的高级用法
#F与Q查询 #*************************** F 查询 ******************** # F 查询数据库中的其他字段!!! #1.查询库存数大于卖出数的书籍 fr ...
- webpack实践(四)- html-webpack-plugin
webpack系列博客中代码均在github上:https://github.com/JEmbrace/webpack-practice <webpack实践(一)- 先入个门> < ...
- Maven pom.xml 全配置(一)常用配置
Maven pom.xml 全配置(一)常用配置 这里贴出一个Maven中出现频率较高的配置参数注释,方便理解项目中Maven的配置具体的作用.如果在此博文中没有找到你想看到的参数,可以移步Maven ...
- linux中文件压缩介绍
原文内容来自于LZ(楼主)的印象笔记,如出现排版异常或图片丢失等问题,可查看当前链接:https://app.yinxiang.com/shard/s17/nl/19391737/1c62bb7f-f ...
- PyTorch-网络的创建,预训练模型的加载
本文是PyTorch使用过程中的的一些总结,有以下内容: 构建网络模型的方法 网络层的遍历 各层参数的遍历 模型的保存与加载 从预训练模型为网络参数赋值 主要涉及到以下函数的使用 add_module ...
- kvm磁盘管理
kvm磁盘管理 kvm虚拟机虚拟磁盘格式转换 各种格式说明介绍 row:裸格式,占用空间较大,不支持快照功能,性能较好,不方便传输(顺序读写) 50G 2G 传输50G qcow2:cow 占用空间小 ...
- mysql与python的交互
mysql是一种关系型数据库,是为了表示事物与事物之间的关系,本身存于数据库中的内容意义并不大,所以广泛应用于编程语言中,python中九含有与MySQL交互的模块 pymysql 编程对mysql的 ...
- CSDN不限积分代下载,知网、万方、sci、IEEE论文代下载,智慧树、超星尔雅刷课
下载内容: 1.CSDN不限积分代下载. 2.知网.万方.sci.IEEE论文代下载. 3.超星尔雅,智慧树刷课. 注:由于本人手抖买一个CSDN会员,想挽回一点损失,所以创立了一个下载群,绝对不是骗 ...
- Appium(十):元素定位(加强版)
1. 元素定位 写完上一篇元素定位的博客,发现实用性基本为零.这几天真的烦死我了,一直在找资料,还去看了一遍appium官网文档.最后结合着selenium的定位方法,测试出几种可行的元素定位方法. ...
- Android Studio中的AndroidManifest.xml文件分析
一.关于AndroidManifest.xml AndroidManifest.xml清单文件是每个Android程序中必须的文件,它是整个Android程序的全局描述文件,除了能声明程序中的Acti ...
