C语言-程序环境和预处理(14.2)

news2025/1/12 9:56:30

目录

预处理详解

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 1000

int 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##y

int 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");
#endif
	return 0;
}

输出 :

输出:hehe

4.2多分支条件编译:

例:

#include <stdio.h>

#define PRINT 1

int main()
{

#if PRINT == 1
	printf("1");
#elif PRINT == 10
	printf("10");
#else 
	printf("???");
#endif 

	return 0;
}

输出:

输出:1
#include <stdio.h>

#define PRINT 10

int main()
{

#if PRINT == 1
	printf("1");
#elif PRINT == 10
	printf("10");
#else 
	printf("???");
#endif 

	return 0;
}

输出:

输出:10
#include <stdio.h>

#define PRINT 100

int main()
{

#if PRINT == 1
	printf("1");
#elif PRINT == 10
	printf("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");
#endif
	return 0;
}

输出:

输出:haha

当然,条件编译也支持嵌套。

在实际中,条件编译也有广泛的应用:

例:

我们可以看一个头文件的源码感受一下:

5. 文件包含

5.1 本地文件包含

先在源文件所在目录下查找,如果该头文件未找到,

编译器就像查找库函数头文件一样在标准位置查找头文件。

5.2 库函数包含

就直接去标准路径下去查找,如果找不到就提示编译错误。

5.3 嵌套文件包含

如果在包含头文件的时候出现这样的情况:

 因为头文件展开后会将所以代码放开,

这样就会造成文件内容的重复。

我们可以用条件编译解决这样的问题:

例:

#ifndef __TEST_H__
#define __TEST_H__
//头文件具体内容:
//...
//
#endif

我们用这个条件编译将头文件的内容包起来,

当再次调用这个头文件的时候,

就会因为 __TEST_H__已经定义过了,而不再编译头文件内容。

当然,如果你嫌麻烦的话,

#pragma once

在头文件中写下这段代码也是同样的效果。

写在最后:

以上就是本篇文章的内容了,感谢你的阅读。

如果喜欢本文的话,欢迎点赞和评论,写下你的见解。

如果想和我一起学习编程,不妨点个关注,我们一起学习,一同成长。

之后我还会输出更多高质量内容,欢迎收看。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/336177.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基础篇:01-微服务概述

1.单体应用与微服务架构区别 如上图左侧为单体应用架构。在传统单体应用中&#xff0c;所有功能模块都在一个工程中编码、部署&#xff0c;即使是集群部署&#xff0c;也只是单体应用的水平复制。 如上图右侧为微服务架构。在微服务架构的项目中&#xff0c;每个应用会按照领域…

浅谈保护数据的加密策略

加密是一种将信息从可读格式转换为混乱字符串的技术。这样做可以防止数据传输中的机密数据泄露。文档、文件、消息和所有其他形式的网络通信都可以加密。加密策略和身份验证服务的结合&#xff0c;还能保障企业机密信息只对授权用户开启访问权限。常见的数据加密包括以下两种&a…

定期备份日志并发送至存储服务器指定路径脚本

根据需求编写一个日志定时备份并发送至存储服务器的脚本定期把三天前的日志文件备份&#xff0c;打包发送至日志备份服务器指定目录&#xff08;修改对应路径拿走即用&#xff09;vim qingli.sh#!/bin/bash#定义星期几week$(date |awk NR1{print $4})num${week}#日志源目录log&…

Android MVI框架搭建与使用

MVI框架搭建与使用前言正文一、创建项目① 配置AndroidManifest.xml② 配置app的build.gradle二、网络请求① 生成数据类② 接口类③ 网络请求工具类三、意图与状态① 创建意图② 创建状态四、ViewModel① 创建存储库② 创建ViewModel③ 创建ViewModel工厂五、UI① 列表适配器②…

【3D目标检测】基于伪雷达点云的单目3D目标检测方法研宄

目录概述细节基准模型点云置信度生成网络特征聚合 DGCNN概述 本文是基于单目图像的3D目标检测方法&#xff0c;是西安电子科技大学的郭鑫宇学长的硕士学位论文。 【2021】【单目图像的3D目标检测方法】 细节 基准模型 作者还是按照伪雷达点云算法的流程设计的&#xff0c;并…

多传感器融合定位十四-基于图优化的定位方法

多传感器融合定位十四-基于图优化的定位方法1. 基于图优化的定位简介1.1 核心思路1.2 定位流程2. 边缘化原理及应用2.1 边缘化原理2.2 从滤波角度理解边缘化3. 基于kitti的实现原理3.1 基于地图定位的滑动窗口模型3.2 边缘化过程4. lio-mapping 介绍4.1 核心思想4.2 具体流程4.…

lamada表达式、stream、collect整理

lamada表达式格式 格式&#xff1a;( parameter-list ) -> { expression-or-statements } 实例&#xff1a;简化匿名内部类的写法 原本写法&#xff1a; public class LamadaTest { public static void main(String[] args) { new Thread(new Runnable() { …

基于PYTHON django四川旅游景点推荐系统

摘 要基于四川旅游景点推荐系统的设计与实现是一个专为四川旅游景点为用户打造的旅游网站。该课题基于网站比较流行的Python 语言系统架构,B/S三层结构模式&#xff0c;通过Maven项目管理工具进行Jar包版本的控制。本系统用户可以发布个人游记&#xff0c;查看景点使用户达到良…

树莓派安装虚拟键盘matchbox-keyboard,解决虚拟键盘乱码问题,解决MIPI DSI触摸屏触控漂移问题

安装虚拟键盘&#xff0c;解决乱码问题 当我们买了触摸屏后&#xff0c;会发现没有键盘&#xff0c;还是无法输入&#xff0c;因此需要虚拟键盘 如果你的语言和地区是中文&#xff0c;那么安装虚拟键盘后可能显示乱码&#xff0c;所以还需要安装中文字体 sudo apt install ttf…

音视频开发—FFMpeg编码解码

FFMpeg 作为音视频领域的开源工具&#xff0c;它几乎可以实现所有针对音视频的处理&#xff0c;本文主要利用 FFMpeg 官方提供的 SDK 实现音视频最简单的几个实例&#xff1a;编码、解码、封装、解封装、转码、缩放以及添加水印。 接下来会由发现问题&#xff0d;&#xff1e;分…

Elasticsearch5.5.1 自定义评分插件开发

文本相似度插件开发&#xff0c;本文基于Elasticsearch5.5.1&#xff0c;Kibana5.5.1 下载地址为&#xff1a; Past Releases of Elastic Stack Software | Elastic 本地启动两个服务后&#xff0c;localhost:5601打开Kibana界面&#xff0c;点击devTools&#xff0c;效果图…

koa ts kick off 搭建项目的基本架子

koa ts kick off 使用ts开发koa项目的基本架子&#xff0c;便于平时随手调研一些技术 项目结构 ├── src │ ├── controller //controller层 │ ├── service //service层 │ ├── routes.ts //路由 │ └── index.ts //项目入…

【图像配准】多图配准/不同特征提取算法/匹配器比较测试

前言 本文首先完成之前专栏前置博文未完成的多图配准拼接任务&#xff0c;其次对不同特征提取器/匹配器效率进行进一步实验探究。 各类算法原理简述 看到有博文[1]指出&#xff0c;在速度方面SIFT<SURF<BRISK<FREAK<ORB&#xff0c;在对有较大模糊的图像配准时&…

04 react css上下浮动动画效果

react css上下浮动动画效果html原生实现上下浮动react 实现上下浮动思路分析实现步骤1.引入useRef2.在所属组件内定义—个变量3.在按钮上添加事件4.定义点击事件对window.scrollTo()进行了解&#xff1a;在react中实现效果图&#xff1a;html原生实现上下浮动 我们有一个导向箭…

【分享】订阅金蝶KIS集简云连接器同步OA付款审批数据至金蝶KIS

方案简介 集简云基于钉钉连接平台完成与钉钉的深度融合&#xff0c;实现钉钉OA审批与数百款办公应用软件&#xff08;如金蝶KIS、用友等&#xff09;的数据互通&#xff0c;让钉钉的OA审批流程与企业内部应用软件的采购、付款、报销、收款、人事管理、售后工单、立项申请等环节…

【2023面试秘籍】 测试工程师的简历该怎么写?

作为软件测试的垂直领域深耕者&#xff0c;面试或者被面试都是常有的事&#xff0c;可是不管是啥&#xff0c;总和简历有着理不清的关系&#xff0c;面试官要通过简历了解面试者的基本信息、过往经历等&#xff0c;面试者希望通过简历把自己最好的一面体现给面试官&#xff0c;…

【Java 面试合集】重写以及重载有什么区别能简单说说嘛

重写以及重载有什么区别能简单说说嘛 前述 这是一道非常基础的面试题&#xff0c;我们在回答的过程中一定要逐一横向比较。 从方法的 修饰符&#xff0c;返回值&#xff0c;方法名&#xff0c;含义&#xff0c;参数等方面进行逐一分析来比较不同。 话不多话&#xff0c;看下…

什么样的台灯适合学生做作业的?开学季,适合孩子写作业的台灯

学生在做作业时&#xff0c;是离不开台灯的&#xff0c;在台灯下学习三四个小时&#xff0c;如果台灯质量不好&#xff0c;那对视力造成很大影响&#xff0c;研究表明&#xff0c;儿童在过亮或者过暗的环境中长时间学习&#xff0c;会导致视力下降等&#xff0c;那么什么样的台…

瀚博半导体载天VA1 加速卡安装过程

背景&#xff1a; 想用 瀚博半导体载天VA1 加速卡 代替 NVIDIA 显卡跑深度学习模型 感谢瀚博的周工帮助解答。 正文&#xff1a; 小心拔出 NVIDIA 显卡&#xff0c;在PCIe 接口插上瀚博半导体载天VA1加速卡&#xff0c;如图&#xff1a; 这时显示屏连接主板的集成显卡 卸载…

cookie和Session的作用和比较

目录 什么是cookie cookie的工作原理 什么是session Session的工作原理 为什么会有session和cookie cookie和session如何配合工作 cookie和Session作用 什么是会话 什么是cookie cookie是web服务器端向我们客户端发送的一块小文件&#xff0c;该文件是干嘛的呢&#xf…