大话C语言:第24篇 预处理

news2025/4/27 16:14:44

1 C语言编译流程

C语言的编译流程包括:

  • 预编译:将.c 中的头文件展开、宏展开,生成的文件是.i 文件。gcc指令:gcc -E file.c -o file.i

  • 编译:将预处理之后的.i 文件生成 .s 汇编文件。gcc指令:gcc -S file.i –o file.s

  • 汇编:将.s 汇编文件生成.o 目标文件。gcc指令:gcc -c file.s -o file.o

  • 链接:将.o 文件链接成目标文件。gcc -o file file.o

2 #include

  • #include<>//用尖括号包含头文件,在系统指定的路径下找头文件

  • #include "" //用双引号包含头文件,先在当前目录下找头文件,找不到,再到系统指定的路径下找。

注意:预处理只是对 include 等预处理操作进行处理并不会进行语法检查;这个阶段有语法错误也不会报错,编译阶段才进行语法检查。

3 #define

#define是用来定义宏,宏是在预编译的时候进行替换。#define包括:

3.1 不带参宏

语法格式:

#define 宏名 常量

例如,定义圆周率,#define PI 3.14,在预编译的时候如果代码中出现了 PI 就用 3.14 去替换。

#include <stdio.h>

#define PI 3.14

int main()
{
	printf("%lf\n",PI);

    return 0;
}

注意,

  • 只要修改宏定义,其他地方在预编译的时候就会重新替换。

  • 宏定义后边不要加分号。

  • 宏定义的作用范围,从定义的地方到本文件末尾。

如果想在中间终止宏的定义范围,可以使用#undef。例如,终止刚定义的PI,#undef PI

#include <stdio.h>

#define PI 3.14

int main()
{
		printf("%lf\n",PI);

#undef PI

#define PI 3.1415

		printf("%lf\n",PI);
	
    return 0;
}

3.2 带参宏

语法格式:

// 表达式是由参数1至参数n构成
#define 宏名(参数1,参数2,...,参数n) 表达式

例如,两数相乘,#define Sum(num1, num2) num1*num2

#include <stdio.h>

#define Sum(num1, num2)  num1*num2

int main(int argc, char *argv[])
{
    int result = Sum(10,20);
    printf("result=%d\n", result);
    
    return 0;
}

实际上,带参数的宏只做简单的参数替换,上述案例中,Sum(10,20),本质上,10*20;如果我们传入的Sum(10+6, 20)的话,宏展开后,10+6 * 20。

#include <stdio.h>

#define Sum(num1, num2)  num1*num2

int main(int argc, char *argv[])
{
    int result = Sum(10+6, 20);
    printf("result=%d\n", result);
    
    return 0;
}

要解决上述问题,最好的办法就是对每个参数单独使用(),改进一下上述代码

#include <stdio.h>

#define Sum(num1, num2)  (num1) * (num2)

int main(int argc, char *argv[])
{
    int result = Sum(10+6, 20);
    printf("result=%d\n", result);
    
    return 0;
}

注意,带参宏被调用多少次就会展开多少次,执行代码的时候没有函数调用的过程,不需要压栈弹栈。所以带参宏,是浪费了空间,因为被展开多次,节省时间。

4 选择性编译

        选择性编译是指在编译过程中,根据特定的条件或环境的不同,选择性地让特定的代码语句有效或无效。这种编译方式允许开发者在不同的环境或配置下编译不同的代码,以实现灵活的程序控制和功能切换。

        在C语言中,选择性编译通常通过预处理器指令来实现,如#ifdef、#ifndef、#else和#endif等。这些指令可以在预处理阶段根据是否定义了某个宏,来决定是否编译特定的代码段。例如,在开发环境中可能需要打印调试信息,而在生产环境中则不需要,通过选择性编译可以在生产环境中排除这些打印语句,以提高程序的运行效率。

        选择性编译的主要作用是防止头文件重复定义和代码编译冗余,从而确保程序的正确性和效率。同时,它也使程序更加灵活和可配置,可以根据不同的需求和环境进行定制。

4.1 #ifdef

语法格式:

#ifdef XXX
	// 功能代码1
#else
	// 功能代码2
#endif

如果已经定义过 XXX(建议取有意义名字) ,就编译功能代码1,否则编译功能代码1。

#include <stdio.h>

#define LANG

int main(int argc, char *argv[])
{
    #ifdef LANG
    	printf("hello world!!\n");
    #else
    	printf("世界,你好!\n");
    #endif
    
    return 0;
}

4.2 #ifndef

语法格式:

#ifndef XXX
	// 功能代码1
#else
	// 功能代码2
#endif

这种方式是和第一种互补,例如

#include <stdio.h>

// #define LANG

int main(int argc, char *argv[])
{
    #ifdef LANG
    	printf("hello world!!\n");
    #else
    	printf("世界,你好!\n");
    #endif
    
    return 0;
}

4.3 #if

语法格式:

#if 表达式
	// 功能代码1
#else
	// 功能代码2
#endif

如果表达式为真,编译功能代码1,否则编译功能代码2。例如

#define DEBUG 1  
  
int main() 
{  
#if DEBUG  
    // 功能代码1: 调试用的代码  
    printf("调试代码.\n");  
#else  
    // 功能代码2: 不包含调试的代码  
    printf("关闭调试代码.\n");  
#endif  
    
    return 0;  
}

注意,如果DEBUG定义为0,表达式为假(false),执行#else。

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

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

相关文章

编译一个叫:未来的IDE-Zed编辑器(Windows平台)

一、前言 截止到2024-6-15&#xff0c;Zed官方并未给出Windows的二进制安装包&#xff0c;如果想在Windows平台使用的话需要自己编译&#xff0c;我是如何编译的请随我道来&#xff0c;有兴趣的码友可以尝试下&#xff0c;在下可不敢保证各位码友按我这方法能100%编译出来&…

流量卡怎么办理的攻略

一、确定需求 在办理流量卡之前&#xff0c;你需要明确自己的需求。根据不同的使用场景&#xff0c;你可能需要考虑以下几个方面&#xff1a; 月租费用&#xff1a;不同运营商提供的流量卡套餐价格各异&#xff0c;从低至19元到高达199元不等。 流量大小&#xff1a;从30GB到3…

CompletableFuture.runAsync的多线程下异步操作

CompletableFuture.runAsync的多线程下异步操作 &#x1f9f0;业务使用场景 ​CompletableFuture.runAsync()​方法是Java中用于创建异步任务的工具&#xff0c;它可以在后台线程中执行指定的任务&#xff0c;并且可以在任务完成后返回结果或执行后续操作。这种方式可以实现多…

Elixir学习笔记——别名、需要、导入和使用

为了便于软件重用&#xff0c;Elixir 提供了三个指令&#xff08;alias、require 和 import&#xff09;以及一个名为 use 的宏&#xff0c;总结如下&#xff1a; # 为模块添加别名&#xff0c;以便可以将其称为 Bar 而不是 Foo.Bar alias Foo.Bar, as: Bar # 需要模块才能使…

数字化校园:打造未来教育新风尚

在21世纪的教育蓝图中&#xff0c;"数字化校园"正逐渐从愿景走向现实&#xff0c;它不仅是科技进步与教育创新深度融合的产物&#xff0c;更是重塑教育生态、引领未来学习风尚的关键力量。随着云计算、大数据、人工智能等前沿技术的蓬勃发展&#xff0c;传统的教育模…

基于RandLA-Net深度学习模型的激光点云语义分割

一、场景要素语义分割部分的文献阅读笔记 RandLA-Net是一种高效、轻量级的神经网络&#xff0c;其可直接逐点推理大规模点云的语义标签。RandLA-Net基于随机点采样获得了显著的计算和内存效率&#xff0c;并采用新的局部特征聚合模块有效地保留了几何细节&#xff0c;弥补了随机…

warning LNK4017: DESCRIPTION 语句不支持目标平台;已忽略

文章目录 warning LNK4017: DESCRIPTION 语句不支持目标平台&#xff1b;已忽略概述笔记备注END warning LNK4017: DESCRIPTION 语句不支持目标平台&#xff1b;已忽略 概述 基于ATL的COM DLL导出函数&#xff0c;无法用__declspec(dllexport)直接在函数上标记为导出函数。 只…

Linux crontabs定时执行任务

文章目录 前言一、安装二、服务1. 启动crond服务2. 关闭crond服务3. 重启crond服务4. 设置crond开机启动5. 禁用crond开机启动6. 查看crond是否开机启动7. 重新载入配置8. 查看crond运行状态 三、使用1. 查看当前用户的crontab2. 编辑用户的crontab3. 删除用户的crontab的内容 …

Linux基础命令[29]-chown

文章目录 1. chown 命令说明2. chown 命令语法3. chown 命令示例3.1 修改属主3.2 修改属组3.3 修改属主和属组3.4 修改文件夹所属 4. 总结 1. chown 命令说明 chown&#xff1a;更改文件的用户或用户组&#xff0c;需要 root 用户或 sudo 权限的用户执行该命令。基本信息如下&…

Elixir学习笔记——模块属性

Elixir 中的模块属性有三个用途&#xff1a; 1.作为模块和函数注释 2.作为编译期间使用的临时模块存储 3.作为编译时常量 让我们检查一下这些示例。 作为注释 Elixir 引入了 Erlang 中的模块属性概念。例如&#xff1a; 在上面的例子中&#xff0c;我们使用模块属性语法定义…

使用GNU Gcov Lcov生成C++单元测试代码覆盖率报告

最近在统计C项目代码的单元测试覆盖率 发现通过gcov和lcov就能将代码单元测试覆盖率定量化和可视化 下面是基于gtest测试框架&#xff0c;对gcov和lcov生成覆盖率的简单示例 工作流程 主要有三个步骤 向GCC编译添加特殊的编译选项以生成可执行文件和*.gcno 运行&#xff08…

Vue3 生命周期函数及其与Vue2的对比总结

Vue3 继续保留了 Vue2 的生命周期钩子&#xff0c;但在 Composition API&#xff08;setup 函数&#xff09;中&#xff0c;它们被改为了一组导入函数。以下是它们的对比&#xff1a; Vue2 生命周期钩子和 Vue3 对应的生命周期函数&#xff1a; 在 Vue3 中&#xff0c;所有的…

TJA1145休眠唤醒调试

目录 项目场景:TJA1145引脚图问题描述SPI链路验证休眠唤醒休眠唤醒配置唤醒报文配置代码相关寄存器:模式寄存器使能CAN唤醒设置唤醒边沿检测事件状态及捕获寄存器CANFD报文不会被识别为有效的唤醒帧项目场景: 最近开发过程中,选择了这颗芯片,踩了很多坑,总算是把这个芯片…

【C语言】解决C语言报错:Undefined Reference

文章目录 简介什么是Undefined ReferenceUndefined Reference的常见原因如何检测和调试Undefined Reference解决Undefined Reference的最佳实践详细实例解析示例1&#xff1a;缺少函数定义示例2&#xff1a;函数声明和定义不匹配示例3&#xff1a;未链接必要的库示例4&#xff…

UWB技术定位系统源码,智慧工厂人员定位系统,独特的射频处理,配合先进的位置算法

UWB技术定位系统源码&#xff0c;高精度人员定位系统源码&#xff0c;智慧工厂人员定位系统源码&#xff0c;室内定位系统源码 本套系统运用UWB定位技术&#xff0c;开发的高精度人员定位系统&#xff0c;通过独特的射频处理&#xff0c;配合先进的位置算法&#xff0c;可以有…

人工智能在风险管理中的创新之路

随着科技的飞速发展&#xff0c;人工智能&#xff08;AI&#xff09;已经渗透到我们生活的方方面面&#xff0c;尤其在风险管理领域&#xff0c;其展现出的巨大潜力令人瞩目。风险管理&#xff0c;作为一个涉及广泛领域的复杂系统&#xff0c;正逐渐依赖于AI技术来提升效率和准…

Linux文件系统【真的很详细】

目录 一.认识磁盘 1.1磁盘的物理结构 1.2磁盘的存储结构 1.3磁盘的逻辑存储结构 二.理解文件系统 2.1如何管理磁盘 2.2如何在磁盘中找到文件 2.3关于文件名 哈喽&#xff0c;大家好。今天我们学习文件系统&#xff0c;我们之前在Linux基础IO中研究的是进程和被打开文件…

将Jar用三种方式生成Windows的安装程序

无论是WEB(spring boot)的JAR,还是JavaFX以及swing的Jar,要生成windows方式。 打包成Windows可执行文件&#xff08;.exe&#xff09;&#xff0c;你可以使用以下三种方法&#xff1a; ### 方法1&#xff1a;使用Inno Setup 1. **构建JavaFX应用程序**&#xff1a; 使用M…

自制一个Linux live固件镜像ISO可引导系统

使用母盘镜像制作两个虚拟&#xff0c;来制作一个包含基本需求的filesystem.squashfs文件&#xff0c;具体看下面的链接 使用的安装镜像 是Linux Mint 制作好的成品 https://cloud.189.cn/t/U32Mvi7FnyA3 &#xff08;访问码&#xff1a;2nbo&#xff09; 最简单制作LIVE CD…

ChatGPT魔法背后的原理:如何做到词语接龙式输出?

介绍 我们都知道 ChatGPT 是 AIGC 工具&#xff0c;其实就是生成式人工智能。大家有没有想过这些问题 &#x1f914;️&#xff1a; 1、我们输入一段话&#xff0c;就可以看见它*噼里啪啦的一顿输出*&#xff0c;那么它的原理到底是什么&#xff1f; 2、到底它是怎么锁定这些…