C语言-程序环境和预处理(14.2)
目录
预处理详解
1.预定义符号
2. #define
2.1 #define定义标识符
2.2 #define 定义宏
2.3 #define 替换规则
注意事项:
2.4 #和##
2.5 带副作用的宏参数
2.6 宏和函数对比
3. #undef
4. 条件编译
4.1 单分支条件编译
4.2 多分支条件编译
4.3 判断是否被定义
5. 文件包含
5.1 本地文件包含
5.2 库函数包含
5.3 嵌套文件包含
写在最后:
预处理详解
思维导图:
1.预定义符号
例:
#include <stdio.h>int main()
{printf("%s\n", __FILE__);//进行编译的文件位置printf("%d\n", __LINE__);//文件当前的行号printf("%s\n", __DATE__);//文件被编译的日期printf("%s\n", __TIME__);//文件被编译的时间return 0;
}
输出:
输出:
F:\my code\c_plus_plus-code-exercise-warehouse\2023_2_8\2023_2_8\test.c
24
Feb 8 2023
19:49:32
注:输出的第一行是我这个文件的路径。
2. #define
2.1 #define定义标识符
#include <stdio.h>#define print printf
#define size sizeof
#define MAX 1000int main()
{print("hello world\n");print("%d\n", size(int));print("%d\n", MAX);return 0;
}
这个其实就是:
1. 用print 代替了 printf,
2. 用size 代替了 sizeof,
3. 用MAX 代替了 1000。
输出:
输出:
hello world
4
1000
另外,建议不要在#define 后面加分号,容易出事。
2.2 #define 定义宏
例:
#include <stdio.h>#define mul(x) ((x)*(x))int main()
{int x = 3;int ret = mul(x);printf("%d\n", ret);return 0;
}
输出:
输出:9
这个的本质其实就是将mul(x) 替换成 ((x)*(x))
当然,x可以是你指定的值。
例:
#include <stdio.h>#define mul(x) ((x)*(x))int main()
{int x = 3;int ret = mul(x);printf("%d\n", ret);ret = mul(3);printf("%d\n", ret);return 0;
}
输出:
输出:
9
9
所以这个也是一样的。
注:
1. 参数列表的左括号必须与mul(你定义的名称)紧邻。
2. 用于对数值表达式进行求值的宏定义建议加上()确保优先级。
2.3 #define 替换规则
1. 在调用宏时,首先对参数进行检查,看看是否包含任何由#define定义的符号,
如果是,它们首先被替换。
2. 替换文本随后被插入到程序中原来文本的位置。
对于宏,参数名被他们的值所替换。
3. 最后,再次对结果文件进行扫描,看看它是否包含任何由#define定义的符号,
如果是,就重复上述处理过程。
注意事项:
1. 宏参数和#define 定义中可以出现其他#define 定义的符号,但是对于宏,不能出现递归。
2. 当预处理器搜索#define定义的符号的时候,字符串常量的内容并不被搜索。
2.4 #和##
例:
#define PRINT(format, x) printf("the value of "#x" is "format"\n", x)int main()
{int a = 10;//printf("the value of a is %d\n", a);PRINT("%d", a);//printf("the value of ""a"" is ""%d""\n", x)int b = 20;//printf("the value of b is %d\n", b);PRINT("%d", b);float f = 3.14f;PRINT("%f", f);//printf("the value of ""f"" is ""%f""\n", x)return 0;
}
输出:
输出:
the value of a is 10
the value of b is 20
the value of f is 3.140000
总结:这里就是使用 # ,把一个宏参数变成对应的字符串。
例2:
#include <stdio.h>#define CAT(x,y) x##yint main()
{int helloworld = 2023;printf("%d\n", CAT(hello, world));return 0;
}
输出:
输出:2023
总结:##可以把位于它两边的符号合成一个符号。
2.5 带副作用的宏参数
例:
#include <stdio.h>#define MAX(x, y) ((x)>(y)?(x):(y))int main()
{int a = 3;int b = 5;int m = MAX(a++, b++);//宏替换之后://((a++)>(b++)?(a++):(b++))printf("%d\n", m);printf("%d %d\n", a, b);return 0;
}
输出:
输出:
6
4 7
因为宏是直接将代码替换下来,所以在使用有副作用的参数时一定要小心。
使用函数的话,就不会出现这样的问题:
例:
#include <stdio.h>int max(int x, int y)
{return x > y ? x : y;
}int main()
{int a = 3;int b = 5;int m = max(a++, b++);printf("%d\n", m);printf("%d %d\n", a, b);return 0;
}
输出:
输出:
5
4 6
因为函数传参就是传的a和b过去计算。
2.6 宏和函数对比
属 性 | #define定义宏 | 函数 |
代 码 长 度 | 每次使用时,宏代码都会被插入到程序中。 除了非常小的宏之外,程序的长度会大幅度增长 | 函数代码只出现于一个地方; 每次使用这个函数时, 都调用那个地方的同一份代码 |
执 行 速 度 | 更快 | 函数调用和返回有额外开销, 所以相对慢一些 |
操 作 符 优 先 级 | 宏参数的求值是: 在所有周围表达式的上下文环境里, 否则邻近操作符的优先级可能会产生 所以建议宏在书写的时候多些括号。 | 函数参数只在函数调用的时候 求值一次, 它的结果值传递给函数。 表达式求值结果更容易预测。 |
带 有 副 作 用 的 参 数 | 参数可能被替换到宏体中的多个位置, 所以带有副作用的参数求值, 可能会产生不可预料的结果。 | 函数参数只在传参的时候 求值一次,结果更容易控制。 |
参 数 类 型 | 宏的参数与类型无关, 只要对参数的操作是合法的, | 函数的参数是与类型有关的, 如果参数的类型不同, 就需要不同的函数, 即使他们执行的任务是 |
调 试 | 宏是不方便调试的 | 函数是可以逐语句调试的 |
递 归 | 宏是不能递归的 | 函数是可以递归的 |
总结:
宏和函数各有优劣,根据实际场景权衡使用。
2.7 命名约定
把宏名全部大写
函数名不要全部大写
3. #undef
这条指令用于移除一个宏定义。
例:
我们发现,移除宏定义MAX之后,再次使用就报错了。
4. 条件编译
我们可以通过使用条件编译根据设定条件屏蔽掉我们不想要的代码。
4.1 单分支条件编译
例:
#include <stdio.h>#define PRINT int main()
{
#ifdef PRINT //还有一个#ifndef是表示PRINT未定义就执行printf("hehe\n");
#endifreturn 0;
}
输出 :
输出:hehe
4.2多分支条件编译:
例:
#include <stdio.h>#define PRINT 1int main()
{#if PRINT == 1printf("1");
#elif PRINT == 10printf("10");
#else printf("???");
#endif return 0;
}
输出:
输出:1
#include <stdio.h>#define PRINT 10int main()
{#if PRINT == 1printf("1");
#elif PRINT == 10printf("10");
#else printf("???");
#endif return 0;
}
输出:
输出:10
#include <stdio.h>#define PRINT 100int main()
{#if PRINT == 1printf("1");
#elif PRINT == 10printf("10");
#else printf("???");
#endif return 0;
}
输出:
输出:???
4.3 判断是否被定义
例:
#include <stdio.h>#define PRINT int main()
{
#if !defined(PRINT)printf("hehe\n");
#endif#if defined(PRINT)printf("haha\n");
#endifreturn 0;
}
输出:
输出:haha
当然,条件编译也支持嵌套。
在实际中,条件编译也有广泛的应用:
例:
我们可以看一个头文件的源码感受一下:
5. 文件包含
5.1 本地文件包含
先在源文件所在目录下查找,如果该头文件未找到,
编译器就像查找库函数头文件一样在标准位置查找头文件。
5.2 库函数包含
就直接去标准路径下去查找,如果找不到就提示编译错误。
5.3 嵌套文件包含
如果在包含头文件的时候出现这样的情况:
因为头文件展开后会将所以代码放开,
这样就会造成文件内容的重复。
我们可以用条件编译解决这样的问题:
例:
#ifndef __TEST_H__
#define __TEST_H__
//头文件具体内容:
//...
//
#endif
我们用这个条件编译将头文件的内容包起来,
当再次调用这个头文件的时候,
就会因为 __TEST_H__已经定义过了,而不再编译头文件内容。
当然,如果你嫌麻烦的话,
#pragma once
在头文件中写下这段代码也是同样的效果。
写在最后:
以上就是本篇文章的内容了,感谢你的阅读。
如果喜欢本文的话,欢迎点赞和评论,写下你的见解。
如果想和我一起学习编程,不妨点个关注,我们一起学习,一同成长。
之后我还会输出更多高质量内容,欢迎收看。
相关文章:
C语言-程序环境和预处理(14.2)
目录 预处理详解 1.预定义符号 2. #define 2.1 #define定义标识符 2.2 #define 定义宏 2.3 #define 替换规则 注意事项: 2.4 #和## 2.5 带副作用的宏参数 2.6 宏和函数对比 3. #undef 4. 条件编译 4.1 单分支条件编译 4.2 多分支条件编译 4.3 判断是…...
VHDL语言基础-时序逻辑电路-计数器
目录 计数器的设计: 计数器的作用: 计数器的实现: 1、用“”函数描述: 用T触发器级联构成的串行进位的二进制加法计数器的仿真波形: 计数器的仿真: 计数器的设计: 计数是一种最简单基本的…...
MySQL数据库07——高级条件查询
前面一章介绍了基础的一个条件的查询,如果多条件,涉及到逻辑运算,and or 之类的。就是高级一点的条件查询。本章来介绍复杂的条件搜索表达式。 AND运算符 AND运算符只有当两边操作数均为True时,最后结果才为True。人们使用AND描述…...
《Terraform 101 从入门到实践》 第四章 States状态管理
《Terraform 101 从入门到实践》这本小册在南瓜慢说官方网站和GitHub两个地方同步更新,书中的示例代码也是放在GitHub上,方便大家参考查看。 军书十二卷,卷卷有爷名。 为什么需要状态管理 Terraform的主要作用是管理云平台上的资源ÿ…...
数据结构之二叉树
🎈一.二叉树相关概念 1.树 树是一种非线性的数据结构,它是由n(n>0)个有限结点组成一个具有层次关系的集合,树结构通常用来存储逻辑关系为 "一对多" 的数据。例如: 关于树的几个重要概念&…...
上海亚商投顾:三大指数集体调整 消费板块逆市活跃
上海亚商投顾前言:无惧大盘涨跌,解密龙虎榜资金,跟踪一线游资和机构资金动向,识别短期热点和强势个股。市场情绪三大指数今日集体调整,沪指全天弱势震荡,创业板指盘中跌超1%。旅游、食品、乳业等大消费板块…...
【2023unity游戏制作-mango的冒险】-开始画面API制作
👨💻个人主页:元宇宙-秩沅 hallo 欢迎 点赞👍 收藏⭐ 留言📝 加关注✅! 本文由 秩沅 原创 收录于专栏:游戏制作 ⭐mango的冒险-开始画面制作⭐ 文章目录⭐mango的冒险-开始画面制作⭐👨&…...
【微服务】Nacos配置管理
🚩本文已收录至专栏:微服务探索之旅 👍希望您能有所收获 Nacos除了可以做配置管理,同样可以当作注册中心来使用。 了解注册中心用法点击跳转👉【微服务】Nacos注册中心 一.引入 当微服务部署的实例越来越多࿰…...
【C++】类与对象理解和学习(上)
专栏放在【C知识总结】,会持续更新,期待支持🌹类是什么?类是对对象进行描述的,是一个模型一样的东西,限定了类有哪些成员,定义出一个类并没有分配实际的内存空间来存储它(实例化后才…...
Pyqt5小案例,界面与逻辑分离的小计算器程序
直接看下最终效果: 使用技术总结 使用Designer设计界面 使用pyuic5命令导出到python文件 新建逻辑处理文件,继承pyuic5导出的文件的类,在里面编写信号与槽的处理逻辑 使用Designer设计界面 要使用Designer,安装一个Python库即…...
leaflet加载KML文件,显示图形(方法2)
第049个 点击查看专栏目录 本示例的目的是介绍演示如何在vue+leaflet中加载KML文件,将图形显示在地图上。 直接复制下面的 vue+openlayers源代码,操作2分钟即可运行实现效果; 注意如果OpenStreetMap无法加载,请加载其他来练习 文章目录 示例效果配置方式示例源代码(共66…...
Mysql 部署 MGR 集群
0. 参考文章 官方文档: MySQL :: MySQL 8.0 Reference Manual :: 18.2 Getting Started 博客: MGR 单主模式部署教程(基于 MySQL 8.0.28) - 墨天轮 (modb.pro) mysql MGR单主模式的搭建 - 墨天轮 (modb.pro) MySQL 5.7 基于…...
迁移至其他美国主机商时需要考虑的因素
网站的可访问性是关系业务的关键因素之一。一个稳定、快速且优化良好的主机上的网站更有可能享受不间断的流量,并在谷歌的SERP中获得更好的排名。因此,在构建企业网站时,选择合适的主机商相当重要。不过就以美国主机为例,由于每个…...
【数据结构】第二章 线性表
文章目录第二章 知识体系2.1 线性表的定义和基本操作2.1.1 线性表的定义2.1.2 线性表的基本操作2.2 线性表的顺序表示2.2.1 顺序表的定义2.2.2 顺序表的基本操作的实现2.3 线性表的链式表示2.3.1 单链表的定义2.3.2 单链表的基本操作实现2.3.3 双链表2.3.4 循环链表2.3.5 静态链…...
RESTful API 为何成为顶流 API 架构风格?
作者孙毅,API7.ai 技术工程师,Apache APISIX Committer 万物互联的世界充满着各式各样的 API ,如何统筹规范 API 至关重要。RESTful API 是目前世界上最流行的 API 架构风格之一,它可以帮助你实现客户端与服务端关注点分离&#x…...
Python基础知识点汇总(列表)
列表的含义 列表由一系列按特定顺序排列的元素组成,是Python中内置的可变序列。 **注:**列表的所有元素放在中括号[]中,相邻的两个元素用逗号分隔; 可将整数、实数、字符串、列表、元组等任何类型的内容放到列表中,且同一列表的元素类型可以不同。 列表的创建和删除 1.…...
新的一年软件测试行业的趋势能够更好?
如果说,2022年对于全世界来说,都是一场极大的挑战的话;那么,2023年绝对是机遇多多的一年。众所周知,随着疫情在全球范围内逐步得到控制,无论是国际还是国内的环境,都会呈现逐步回升的趋势&#…...
Threejs中的Shadow Mapping(阴影贴图)
简而言之,步骤如下: 1.从灯光位置视点(阴影相机)创建深度图。 2.从相机的位置角度进行屏幕渲染,在每个像素点,比较由阴影相机的MVP矩阵计算的深度值和深度图的值的大小,如果深度图值小的话&…...
本质安全设备标准(IEC60079-11)的理解(四)
本质安全设备标准(IEC60079-11)的理解(四) 对于标准中“Separation”的理解 IEC60079-11使用了较长的篇幅来说明设计中需要考虑到的各种间距, 这也从一定程度上说明了间距比较重要,在设计中是需要认真考虑…...
(record)QEMU安装最小linux系统——TinyCore(命令行版)
文章目录QEMU安装最小linux系统——TinyCore参考QEMU使用qemu创建tinycore虚拟机再次启动文件保存QEMU安装最小linux系统——TinyCore 简单记录安装过程和记录点 参考 [原创] qemu 与 Tiny Core tinycore的探索 QEMU qemu不多介绍,这里是在WSL2上安装的linux版…...
C++中的cast类型转换
reinterpret_cast用法:reinpreter_cast<type-id> (expression)type-id必须是一个指针、引用、算术类型、函数指针或者成员指针。它可以把一个指针转换成一个整数,也可以把一个整数转换成一个指针。这个操作符能够在非相关的类型之间转换。操作结果…...
西瓜数据集读取的详细解决方案
大家好,我是爱编程的喵喵。双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。喜欢通过博客创作的方式对所学的知识进行总结与归纳,不仅形成深入且独到的理…...
Mac开发环境配置
一、mac 安装homebrew 1. 必要性 homebrew可以通过bash命令快速安装配置开发环境,并且在大多数情况下可以实现环境的自动配置。(一键安装配置) 2. 收益 节省开发环境工具配置时间,提高人效。 3. 安装步骤 打开mac终端…...
概率论面试题1:玫瑰花
概率论面试题 1. 一个活动,n个女生手里拿着长短不一的玫瑰花,无序的排成一排,一个男生从头走到尾,试图拿更长的玫瑰花,一旦拿了一朵就不能再拿其他的,错过了就不能回头,问最好的策略࿱…...
【DGL】图分类
目录概述数据集定义Data LoaderDGL中的batched graph定义模型训练参考概述 除了节点级别的问题——节点分类、边级别的问题——链接预测之外,还有整个图级别的问题——图分类。经过聚合、传递消息得到节点和边的新的表征后,映射得到整个图的表征。 数据…...
时间复杂度的计算(2023-02-10)
时间复杂度的计算 时间复杂度的计算分为三大类:一层循环、二层循环和多层循环。 一层循环 1.找出循环趟数t及每轮循环i的变化值 2.确立循环停止的条件 3.得出t与i之间的关系 4.联立两式,得出结果 eg: void fun(int n) {int i0;while (i*i*i<n)i;…...
测试开发之Django实战示例 第六章 追踪用户行为
第六章 追踪用户行为在之前的章节里完成了小书签将外站图片保存至本站的功能,并且实现了通过jQuery发送AJAX请求,让用户可以对图片进行喜欢/不喜欢操作。这一章将学习如何创建一个用户关注系统和创建用户行为流数据,还将学习Django的信号框架…...
红米9a手动root方法
简介 已知红米6A/6/9/9A/9C/10A机器都可以快速解锁BL,无任何变砖风险 并且秒解锁BL后和官方解锁一样,无任何其他不良影响。推荐大家使用官网解锁,需要等待7天。 BootLoader BootLoader是在操作系统内核运行之前运行的一段小程序。其实…...
Open3D 点云最小二乘法拟合平面(剔除噪声,Python版本)
除了诱惑之外,我可以抵抗任何事物。 ----王尔德 文章目录 一、简介二、实现代码三、实现效果参考资料一、简介 这个算法的思路很简单,就是通过剔除一些异常点来拟合更为合适的平面,具体过程如下所示: 1、首先使用最小二乘法拟合一个平面系数的初值。 2、计算所有有效点到拟合…...
【SpringBoot】简述springboot项目启动数据加载内存中的三种方法
一、前言一般来说,SpringBoot工程环境配置放在properties文件中,启动的时候将工程中的properties/yaml文件的配置项加载到内存中。但这种方式改配置项的时候,需要重新编译部署,考虑到这种因素,今天介绍将配置项存到数据…...
中国建设工程信息网清欠/深圳seo优化排名
K的因子中只包含2 3 5。满足条件的前10个数是:2,3,4,5,6,8,9,10,12,15。所有这样的K组成了一个序列S,现在给出一个数n,求S中 > 给定数的最小的数。例如:n 13,S中 > 13的最小的数是15,所以输出15。In…...
icp备案查询站长工具/seo的作用
网络与通信 N e t w o r k&C o mmu n i c a t i o n计算机网络技术的发展模式研究文/李祥龙研究、探索、试验以及考证,也伴随着新兴技2计算机网络技术快速发展模式分析计算机技术的发展模式对一个国家、地区和社会的成长有着十分重要的影响,其制度、…...
wordpress删掉不需要的/seo短视频网页入口营销
什么是指针 是一种特殊的数据类型,使用它可以定义指针变量,指针变量储存的是整型数据,代表了内存的编号,通过这个标号可以访问对应的内存。 为什么要使用指针 1、函数之间相互独立,但有时候需要共享变量 …...
广告制作公司开票范围/百度seo排名优化如何
安装说明 下载Pixymon和pixy对应markone的固件,在这里。 固件必须是firmware_IRLOCKpixy_1.0.1.hex irlock markone 然后固件里 irlock pixy 给Pixy刷固件的方法 打开PixyMon,将Pixy连接到电脑,然后长按Pixy上的按钮,进入烧…...
做app和做网站/广州网络推广平台
公元二零一二年,此猴子在此国度出世。掌握内功心法为传说中的特级心法《道难特》另外了解过另一特级心法为《甲骨文之佳万》。为此在纠结以后的拜师之路,请各位大侠帮助,谢谢。转载于:https://www.cnblogs.com/ja-net/archive/2012/05/10/249…...
iis网站开发/网推平台有哪些比较好
快捷键:Ctrl Alt s,进入Settings >Editor>Live Templates>output中添加一个项,选择第一个Live Template 点击Define,勾选Everywhere后确定才能在自己想要的地方快速调用出来: 如下,在xml中一样…...