深度解析C语言——预处理详解

news/2024/7/26 11:18:42/文章来源:https://blog.csdn.net/2302_77644537/article/details/137276178

        对C语言有一定了解的同学,相信对预处理一定不会陌生。今天我们就来聊一聊一些预处理的相关知识。预处理是在编译之前对源文件进行简单加工的过程,主要是处理以#开头的命令,例如#include <stdio.h>、#define等。预处理是C语言的一个重要功能,在预处理阶段完成。当对一个源文件进行编译时,系统将自动调用预处理程序对源程序中的预处理部分作处理,处理完毕自动进入对源程序的编译。

~~~正文开始~~~

预定义符号

        什么是预定义符号:预定义符号是由编译器预先设置好的特殊标识符,它们代表了特定的信息,如编译器版本、目标平台信息、编译选项等。在C语言中, 也设置了一些预定符号,可以直接使用。

//常见的C语言预定义符号
__FILE__ //进⾏编译的源⽂件
__LINE__ //⽂件当前的⾏号
__DATE__ //⽂件被编译的⽇期
__TIME__ //⽂件被编译的时间

使用举例:

int main()
{printf("file:%s line:%d\n", __FILE__, __LINE__);//输出结果:file:C:\Users\test.c line:269printf("date:%s time:%s\n", __DATE__, __TIME__);//输出结果:date:Apr  2 2024 time:17:04:04return 0;
}

#define

#define定义常量

//基本语法
#define name stuff

 使用举例:

//最常见的定义方式
#define MAX 1000
//为register这个关键字,创建⼀个简短的名字。有点类似typedef
#define reg register 
//⽤更形象的符号来替换⼀种实现(死循环)
#define do_forever for(;;) 
//在写case语句的时候⾃动把 break写上。
#define CASE break;case 
//如果定义的 stuff过⻓,可以分成⼏⾏写,除了最后⼀⾏外,每⾏的后⾯都加⼀个反斜杠(续⾏符)。
#define DEBUG_PRINT printf("file:%s\tline:%d\t \date:%s\ttime:%s\n" ,\__FILE__,__LINE__ , \__DATE__,__FILE__)

       思考一下,为什么在define定义标识符的时候,后面不加分号呢?我们知道define定义的标识符在预处理阶段就会被替换,如果加上分号,就可能导致程序出错。比如:

#define MAX 10;
int main()
{//替换之后:printf("%d\n", 10;);//就会有语法错误printf("%d\n", MAX);return 0;
}

#define定义宏 

#define 机制有⼀个规定,允许把参数替换到⽂本中,这种实现通常称为宏(macro)或定义宏(define macro)。

//宏的申明⽅式:
#define name( parament-list ) stuff

注意:参数列表的左括号必须与name紧邻,如果两者之间有任何空白存在,参数列表就会被解释为stuff的一部分。

  使用举例:

#define MUL(x) x * x
int main()
{printf("%d\n", MUL(5));//输出:25return 0;
}

上面的代码看上去是不是非常完美?实际上存在了一个特别大的bug,请看下面一段代码:

#define MUL(x) x * x
int main()
{printf("%d\n", MUL(5 + 1));return 0;
}

这段代码的结果是多少呢?36?no no no,实际上上面的代码会被替换成:

printf("%d\n", 5 + 1 * 5 + 1);//结果为11

所以我们在使用宏的时候一定要注意,应该把上面代码修改为:

#define MUL(x) ((x) * (x))

这样就可以得到我们想要的结果了,所以用于对数值表达式进行求值的宏定义都应该用这种方式加上括号,避免在使用宏时由于参数中的操作符或邻近操作符之间不可预料的相互作用


宏与函数的对比

宏通常被应用于执行简单的运算。

和函数相比宏的优势

比如在两个数中找出较大的一个时,写成下面的宏,更有优势一些。

#define MAX(a, b) ((a)>(b)?(a):(b))

 优势有二:

  1. 用于调用函数和从函数返回的代码可能比实际执行这个小型计算工作所需要的时间更多。所以宏比函数在程序的规模和速度方面更胜一筹。
  2. 更为重要的是函数的参数必须声明为特定的类型。所以函数只能在类型合适的表达式上使用。反之这个宏可以适用于整形、长整型、浮点型等类型。宏是类型无关的。

和函数相比宏的劣势

  1. 每次使用宏的时候,一份宏定义的代码将插入到程序中。除非宏比较短,否则可能大幅度增加程序的长度。
  2. 宏是没法调试的。
  3. 宏由于类型无关,也就不够严谨。
  4. 宏可能会带来运算符优先级的问题,导致程容易出现错

宏和函数的对比

 命名约定

⼀般来讲函数的宏的使用语法很相似。所以语言本身没法帮我们区分二者。那我们平时的用个习惯是:

把宏名全部大写

函数名不要全部大写

但是也有例外,offsetof就是一个宏,但它却是全部小写

#undef

这条指令用于移除一个宏定义。

  使用举例:

#define MAX 20
int main()
{   printf("%d\n", MAX);
#undef MAX  //移除宏定义//printf("%d\n", MAX);  //error//也可以再次定义宏
#define MIN 10printf("%d\n",MIN);return 0;
}

 条件编译

        在编译⼀个程序的时候我们如果要将一条语句(一组语句)编译或者放弃是很方便的。因为我们有条件编译指令。比如说:调试性的代码,辛辛苦苦写的删除可惜,保留又碍事,所以我们可以选择性的编译。

  使用举例:

#define __DEBUG__
int main()
{int arr[10] = { 0 };for (int i = 0; i < 10; i++){arr[i] = i;#ifdef __DEBUG__       //若为真,则执行printf语句printf("%d\n", arr[i]);//为了观察数组是否赋值成功。#endif //__DEBUG__}return 0;
}

 常见的条件编译指令

  • 条件编译
#if 常量表达式
//...
#endif
//常量表达式由预处理器求值。
//例:
#define __DEBUG__ 1
#if __DEBUG__
//..
#endif
  • 多个分支的条件编译
#if 常量表达式
//...
#elif 常量表达式
//...
#else
//...
#endif
  •  判断是否被定义
#if defined(symbol)
#ifdef symbol#if !defined(symbol)
#ifndef symbol

 使用举例:

#define __DEBUG__ 
int main()
{#if defined(__DEBUG__)printf("haha\n");#endif#if !defined(__DEBUG__)printf("haha\n");#endif//打印结果:hahareturn 0;
}
  • 嵌套指令
#if defined(OS_UNIX)#ifdef OPTION1unix_version_option1();#endif#ifdef OPTION2unix_version_option2();#endif
#elif defined(OS_MSDOS)#ifdef OPTION2msdos_version_option2();#endif
#endif

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

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Element Plus:图标

第一步&#xff1a;npm install element-plus/icons-vue 第二步&#xff1a;在main.js里 import * as ElementPlusIconsVue from element-plus/icons-vueconst app createApp(App) for (const [key, component] of Object.entries(ElementPlusIconsVue)) {app.component(key,…

HarmonyOS 应用开发之通过键值型数据库实现数据持久化

场景介绍 键值型数据库存储键值对形式的数据&#xff0c;当需要存储的数据没有复杂的关系模型&#xff0c;比如存储商品名称及对应价格、员工工号及今日是否已出勤等&#xff0c;由于数据复杂度低&#xff0c;更容易兼容不同数据库版本和设备类型&#xff0c;因此推荐使用键值…

ubuntu22.04@Jetson Orin Nano安装配置VNC服务端

ubuntu22.04Jetson Orin Nano安装&配置VNC服务端 1. 源由2. 环境3. VNC安装Step 1: update and install xserver-xorg-video-dummyStep 2: Create config for dummy virtual displayStep3: Add the following contents in xorg.conf.dummyStep 4: Update /etc/X11/xorg.con…

【详解】运算放大器工作原理及其在信号处理中的核心作用

什么是运算放大器 运算放大器&#xff08;简称“运放”&#xff09;是一种放大倍数非常高的电路单元。在实际电路中&#xff0c;它常常与反馈网络一起组成一定的功能模块。它是一种带有特殊耦合电路和反馈的放大器。输出信号可以是输入信号的加法、减法、微分和积分等数学运算…

c++对象指针

对象指针在使用之前必须先进行初始化。可以让它指向一个已定义的对象&#xff0c;也可以用new运算符动态建立堆对象。 定义对象指针的格式为&#xff1a; 类名 *对象指针 &对象; //或者 类名 *对象指针 new 类名(参数); 用对象指针访问对象数据成员的格式为&#xff1a…

Django详细教程(二) - 部门用户管理案例

文章目录 前言一、新建项目二、新建app三、设计表结构四、新建数据库五、新建静态文件六、部门管理1.部门展示2.部门添加3.部门删除4.部门编辑 七、模板继承八、用户管理1.辨析三种方法方法一&#xff1a;原始方法方法二&#xff1a;Form组件(简便)方法三&#xff1a;ModelForm…

1.Netty介绍及NIO三大组件

Netty网络编程Netty的底层是NIO&#xff08;非阻塞IO&#xff09;&#xff0c;常用的多线程和线程池使用的是阻塞IO&#xff0c;其效率并不高。支持高并发&#xff0c;性能好高性能的服务端程序、客户端程序 NIO三大组件 一、Channel 读写数据的双向传输通道 常见的传输通道…

【C++第二阶段】继承多态电脑组装实例

你好你好&#xff01; 以下内容仅为当前认识&#xff0c;可能有不足之处&#xff0c;欢迎讨论&#xff01; 文章目录 继承继承语法继承方式继承中的对象模型继承中构造和析构顺序同名成员处理同名静态成员处理多继承语法菱形继承问题 多态多态基本概念重写&重载 多态原理多…

linux基础命令篇: centos7虚拟机网络配置——NAT模

linux基础命令篇&#xff1a; centos7虚拟机网络配置——NAT模式 1搞清楚NAT模式概念 在网络地址转换&#xff08;NAT&#xff09;模式下&#xff0c;虚拟机与宿主机共享一个IP地址。虚拟机的所有网络流量都会通过宿主机的IP地址进行转换&#xff0c;然后发送到外部网络。这意…

【HTML】标签学习(下.2)

&#xff08;大家好哇&#xff0c;今天我们将继续来学习HTML&#xff08;下.2&#xff09;的相关知识&#xff0c;大家可以在评论区进行互动答疑哦~加油&#xff01;&#x1f495;&#xff09; 目录 二.列表标签 2.1 无序列表(重点) 2.2有序列表(理解) 2.3 自定义列表(重点…

基于多数据源融合的医疗知识图谱框架构建研究

基于多数据源融合的医疗知识图谱框架构建研究 提出背景医学数据源医学数据获取方法知识图谱的构建 提出背景 论文&#xff1a;基于多数据源融合的医疗知识图谱框架构建研究 本文以医疗领域的实际应用需求为出发点&#xff0c;从医疗大数据获取、医疗实体及关系标注、医疗实体…

怎么加密文件夹?文件夹加密软件有哪些?

文件夹加密是保护文件夹数据安全的重要手段&#xff0c;可以有效地避免文件夹数据泄露。那么&#xff0c;文件夹加密软件有哪些呢&#xff1f;下面我们就一起来了解一下吧。 文件夹加密超级大师 文件夹加密超级大师作为一款专业的文件夹加密软件&#xff0c;支持五种文件夹加密…

网页实现-基于深度学习的车型识别与计数系统(YOLOv8/v7/v6/v5代码+训练数据集)

摘要&#xff1a;本文深入研究了基于YOLOv8/v7/v6/v5的车型识别与计数&#xff0c;核心采用YOLOv8并整合了YOLOv7、YOLOv6、YOLOv5算法&#xff0c;进行性能指标对比&#xff1b;详述了国内外研究现状、数据集处理、算法原理、模型构建与训练代码&#xff0c;及基于Streamlit的…

Java字符串、集合的基本使用

一、字符串 1.构造字符串 使用直接赋值获取一个字符串对象 String s1 "abc"; 使用new的方法获取一个字符串对象 //空参构造&#xff1a;获取空白的字符串对象 String s2 new String();//带参数的构造 String s3 new String("abc"); 传递一个字符数组&am…

深入核心招聘场景,用友大易帮助健合集团解决「渠道、效率、体验」三件事

自1999年成立以来&#xff0c;健合集团一直致力于婴幼儿营养与护理、成人自然健康营养与护理、以及宠物营养与护理三大核心领域。作为全球高端家庭营养及护理品牌的佼佼者&#xff0c;健合集团始终秉持「让人们更健康更快乐」的企业理念&#xff0c;这不仅体现在产品和服务上&a…

MP4文件中h264的 SPS、PPS获取

MP4文件中h264的SPS、PPS获取 如下图所示&#xff0c;为avcC 1 【参考依据】ISO/IEC 14496-15 2 【综述】在H264中&#xff0c;SPS和PPS存在于NALU header中&#xff0c;而在MP4文件中&#xff0c;SPS和PPS存在于AVCDecoderConfigurationRecord&#xff0c; 首先要定位avcC. …

HTML1:html基础

HTML 冯诺依曼体系结构 运算器 控制器 存储器 输入设备 输出设备 c/s(client客户端) 客户端架构软件 需要安装,更新麻烦,不跨平台 b/s(browser浏览器) 网页架构软件 无需安装,无需更新,可跨平台 浏览器 浏览器内核: 处理浏览器得到的各种资源 网页: 结构 HTML(超…

Redis 全景图(1)--- 关于 Redis 的6大模块

这是我第一次尝试以长文的形式写一篇 Redis 的总结文章。这篇文章我想写很久了&#xff0c;只是一直碍于我对 Redis 的掌握没有那么的好&#xff0c;因此迟迟未动笔。这几天&#xff0c;我一直在看各种不同类型的 Redis 文章&#xff0c;通过阅读这些文章&#xff0c;引发了我对…

zabbix主动发现,注册及分布式监控

主动发现 结果 主动注册 结果 分布式监控 服务机&#xff1a;132 代理机&#xff1a;133 客户端&#xff1a;135 代理机 数据库赋权&#xff1a; 代理机配置 网页上配置代理 客户端配置 网页上配置主机 重启代理机服务 网页效果

算法学习——LeetCode力扣图论篇1(797. 所有可能的路径、200. 岛屿数量、695. 岛屿的最大面积)

算法学习——LeetCode力扣图论篇1 797. 所有可能的路径 797. 所有可能的路径 - 力扣&#xff08;LeetCode&#xff09; 描述 给你一个有 n 个节点的 有向无环图&#xff08;DAG&#xff09;&#xff0c;请你找出所有从节点 0 到节点 n-1 的路径并输出&#xff08;不要求按特…