C语言最终讲:预处理详解

C语言最终讲:预处理详解

  • 1.预定义符号
  • 2.#define定义常量
  • 3.#define定义宏
  • 4.带有副作用的宏参数
  • 5.宏替换的规则
  • 6.宏和函数的对比
    • 6.1宏的优势
      • 6.1.1\符号
    • 6.2宏的劣势
  • 7.#和##
    • 7.1#运算符
    • 7.2##运算符
  • 8.命名约定
  • 9.#undef
  • 10.命令行定义
  • 11.条件编译
  • 12.头文件的包含
    • 12.1本地文件包含
    • 12.2库文件包含
    • 12.3嵌套文件包含
  • 13.其他预处理指令

结语:这一讲是C语言基础知识的最后一讲了,后续将会学习数据结构相关的知识,坚持不易,希望各位都能坚持在自己所干的事情上,我们共勉

这一讲讲的是预处理相关的内容,上一讲虽然我们已经了解了一些关于#define相关的知识,这一讲我们讲详细阐述它的作用和缺陷

1.预定义符号

C语言中有着一些预定义符号,可以直接使用,是在预处理阶段进行处理的:

__FILE__ //进⾏编译的源⽂件
__LINE__ //⽂件当前的⾏号
__DATE__ //⽂件被编译的⽇期
__TIME__ //⽂件被编译的时间
__STDC__ //如果编译器遵循ANSI C,其值为1,否则未定义

它们的使用方法为:

int main()
{printf("%s\n", __FILE__);//打印当前文件所在的位置printf("%d\n", __LINE__);//打印行号printf("%s\n", __DATE__);//打印日期printf("%s\n", __TIME__);//打印时间//printf("%d\n", __STDC__);//err,表明VS并不完全遵循ANSI C标准return 0;
}

运行结果如下:
在这里插入图片描述

2.#define定义常量

我们知道,define定义的常量是直接替换的,所以有着下面的几处用法:

#define TEST 20 //方法1,直接定义常量
#define I int//方法2:可以为关键字替换成一个更简便的名字
#define CASE break;case//方法3:这是一种很“奇葩”的用法,建议还是别用了int main()
{//使用起来也很方便,方法1:int a = TEST;//直接使用即可//方法2:I b = 20;printf("%d\n", b);//方法3:int input = 1;scanf("%d", &input);switch (input){case 1:CASE 2 ://这样后面就不用加break了CASE 3 :break;}return 0;
}

对于加;问题,解答如下:

#define ROW 20;//假设#define定义时加上了;int main()
{int a = ROW;//这是a就会被替换成int a = 20;;这是后面就会有两个;,特殊情况下会出现错误//总结:加上;可能会出现错误,不加;肯定不会出现错误,所以还是不要加上;return 0;
}

3.#define定义宏

定义方式:

#define name( parament-list ) stuff
name为宏名称
parament-list是由逗号隔开的符号表,可能出现在stuff中
stuff可以是一个计算方式,也可以是一个指令等等
//需要注意的是,符号表的左括号必须要和name紧密相连,否则括号里的内容也会被当成是stuff中的一部分

使用实例:

#define MUL(x) x*x//定义了一个宏,实现x的平方int main()
{int a = 5;int b = MUL(a);printf("b = %d\n", b);//结果为25return 0;
}

看起来使用起来很好用,但是我们也要注意符号的优先级问题,如下:

#define ADD(x) x+xint main()
{int a = 5;int b = ADD(a) * ADD(a);printf("%d\n", b);//我们可能会误以为结果为10 * 10,这不就是100吗,但是并非如此//b会被替换成int b = 5+ 5*5 +5,根据符号优先级问题,结果为35return 0;
}

知道错误就要改正,所以改正方法如下:

#define ADD(x) ((x)+(x)) //在x上加一个括号,因为x为表达式,如果传入5+5,也要被括起来//总体上再加上一个括号,是为了将它们算成一个整体int main()
{int a = 5;int b = ADD(a) * ADD(a);printf("%d\n", b);//这样才能计算出一个整体的值,此时结果为100return 0;
}

所以,对于宏的定义,我们不能吝惜括号

4.带有副作用的宏参数

什么是带有副作用呢,其实就是表达式求值会改变原来的参数的值,如下:

x + 1;//不带副作⽤
x++;//带有副作⽤

为了验证此写法的危害性,我们举例来说明:

#define MAX(a, b) ( (a) > (b) ? (a) : (b) )
int main()
{int x = 5;int y = 8;int z = MAX(x++, y++);//这里会被替换成((x++)>(y++) ? (x++):(y++))//先看问号左边,后置++,先使用后++,5>8为假,所以看的是y++的值,还是先使用后++,但是要注意的是//由于++,此时y的值已经变成了9,所以z被赋值为9,y再++,值位10//而x只++了一次,因为第二次只对y进行了++,所以x的值为6printf("x=%d y=%d z=%d\n", x, y, z);//输出的结果是什么?//所以输出的结果应该为6 10 9return 0;
}

5.宏替换的规则

1.在调用宏时,首先看参数中是否包含由#define定义的符号,如果有,它们首先被替换:

#define R 2
#define RR R+2
//其中RR中包含了一个常量的定义R,所以R先被替换成2,再计算RR

2.替换文本随后会被插入到原来文本所在的位置
3.最后还会对文件进行扫描,看是否仍然存在#define定义的符号,如果有,继续进行上面的两个步骤

对于宏,需要注意的是,它并不能出现递归

6.宏和函数的对比

上面我们了解了宏,那么既然宏这么好用,那我们是不是就能够将宏代替为函数使用呢,肯定是不能的,下面我们来分析宏相较于函数的利与弊

6.1宏的优势

1.对于简单的任务,宏有着天然的优势:

#define MAX(a, b) ((a)>(b)?(a):(b))
1.首先,宏的参数是类型无关的,也就是说,宏参数能够接受任意类型的值,在特定条件下是很好用的
2.其次,宏在使用过程中也是一个简单的替换,函数不同,函数在使用前后会产生空间的创建和销毁,使用效率较慢

2.相较于函数,宏有着函数不能够实现的情况:

//如果我们要使用函数进行空间的开辟:
int* Mal(int a, int size)
{//我们首先会将返回值和参数确定,也就是说,它们的类型都是固定的,用于整形开辟的函数就不能用于浮点型类型的开辟return (int*)malloc(a * sizeof(int));
}//但是当我们使用宏时:
#define MAL(num, type)  \(type*)malloc(num * sizeof(type))
int main()
{//使用函数开辟空间int* pa = Mal(10, sizeof(int));//假设我们要开辟10个整形的空间//使用宏开辟空间int* pb = MAL(10, int);//1.我们可以将int直接作为参数传入//2.此时宏的使用包含的情况更多,更好用return 0;
}

6.1.1\符号

上面我们使用了一个\符号,我们看看这是怎么个事:

#define MAL(num, type)  \(type*)malloc(num * sizeof(type))
//这个宏定义中使用了\符号,它是一个换行符,当宏里的内容过多时,就可以在后边加上一个\符号进行换行
//但是要注意的是,\后边什么也不能有,包括空格
//可以理解为:一个\抵消了一个\n,所以后边不能跟任何东西

6.2宏的劣势

1.每次使用宏时,一份宏代码就会被插入到程序中,如果宏很长的话,会大幅增加程序的长度
2.宏在预处理阶段就被替换,不能够被调试
3.宏由于类型无关,所以也容易出现问题
4.宏可能造成运算符优先级的问题

对于宏和函数的一个对比,我们可以看图:
图不是很好,大家对付着看
在这里插入图片描述

7.#和##

7.1#运算符

#所实现的操作被称为“字符串化”,作用是将宏的一个参数转换成字符串变量:

#define PRINT(n) printf("The value of " #n " is %d", n);int main()
{int a = 10;PRINT(a);//此时会打印出:The value of a is 10//预处理阶段,代码会被替换成:printf("the value of ""a" " is %d", a);//也就是说,#n就表示"n",对于printf,多个""会被合并成一个return 0;
}

7.2##运算符

该运算符被称为记号粘合,可以将两边的符号合并成一个符号

使用方法如下:

//当我们要求两个数的最大值时:
//使用函数完成较为繁琐,因为不同类型需要分别进行处理:
int int_max(int x, int y)
{return x > y ? x : y;
}float float_max(float x, float y)
{return x > y ? x:y;
}//我们可以使用宏来实现:
#define MAX(type)                      \type type##_max(type x, type y)\{                              \return x > y ? x : y;      \}MAX(int);//这时我们可以依靠宏来创建一个函数,函数名为int_maxint main()
{//使用方法也比较简单:int ret1 = int_max(2, 3);//我们直接使用创建好的函数即可printf("%d\n", ret1);return 0;
}

8.命名约定

在命名宏和函数时,有着一个不成文的约定:

1.把宏名全部大写
2.函数名不要全部大写

9.#undef

用于移除一个宏定义,使用方法如下:

#define A 20int main()
{int a = A;printf("%d\n", a);//20#undef Aint b = A;//errreturn 0;
}

10.命令行定义

在不同的场景下,我们定义的数组长度需求可能不同,所以许多C的编译器提供了一种命令行定义的能力,允许在命令行中,也就是使用前对一些变量进行赋值

例子:

int main()
{int array[ARRAY_SIZE];int i = 0;for (i = 0; i < ARRAY_SIZE; i++){array[i] = i;}for (i = 0; i < ARRAY_SIZE; i++){printf("%d ", array[i]);}printf("\n");return 0;
}

编译指令:

//linux 环境演⽰
gcc - D ARRAY_SIZE = 10 programe.c
这里表示给ARRAY_SIZE赋值为10,然后再执行程序

11.条件编译

对于一些为了调试而使用的代码,删除了可惜,保留了又费事,所以我们可以通过条件编译来选择性编译

常见的条件编译指令:

1.
#if 常量表达式//...
#endif
//常量表达式由预处理器求值。
如:
#define __DEBUG__ 1
#if __DEBUG__//..
#endif2.多个分⽀的条件编译
#if 常量表达式//...
#elif 常量表达式//...
#else//...
#endif3.判断是否被定义
#if defined(symbol)
#ifdef symbol#if !defined(symbol)
#ifndef symbol4.嵌套指令
#if defined(OS_UNIX)#ifdef OPTION1unix_version_option1();#endif#ifdef OPTION2unix_version_option2();#endif
#elif defined(OS_MSDOS)#ifdef OPTION2msdos_version_option2();#endif
#endif

总结:

1.#if#endif是一体的,它们两个必须同时使用
2.#if#elif就相当于if else
3.#if defined(symbol)表示如果定义了symbol,也可以写成:#ifdef symbol
4.#if !defined(symbol)表示如果没有定义symbol,也可以写成:#ifndef symbol

它的原理为:

//使用原理为:
//当条件为真时,保留代码
//条件为假时,删除代码
int main()
{
#if 1printf("haha\n");//条件为真,保留代码
#elif 0printf("hehe\n");//条件为假,删除代码
#endifreturn 0;
}
//所以上面的代码在预处理之后就变成了:
int main()
{printf("haha\n");//条件为真,保留代码return 0;
}

12.头文件的包含

头文件有两种包含方式:本地文件包含和库文件包含,它们的包含方式是什么呢,我们来看:

12.1本地文件包含

#include "filename"
使用“”来包含

查找策略:

1.现在源文件目录下查找,也就是这个路径:

在这里插入图片描述

2.如果没有找到,在库函数头文件中进行查找
3.如果找不到,提示编译错误

Linux环境下标准头文件的路径:

/usr/include

VS环境下(现在VS环境下头文件的路径应该会有差异,这里只当成一个代表):

C:\Program Files (x86)\Microsoft Visual Studio 12.0\VC\include
//这是VS2013的默认路径

12.2库文件包含

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

那么我们就要问了,是不是库文件的包含也可以使用""呢,原理上来说可以,但是缺点也很明显:
1.效率低
2.这样就不能够分辨出哪个是包含的是标准库文件,哪个是包含本地文件

12.3嵌套文件包含

当我们使用#include包含多个头文件时,会发生什么呢:

在这里插入图片描述
在预处理阶段,竟然会有5个头文件被写入了.c文件中,因为头文件的包含其实就是将头文件的内容写入.c文件中,情况如下:

struct Stu
{int id;char name[20];
};
struct Stu
{int id;char name[20];
};
struct Stu
{int id;char name[20];
};
struct Stu
{int id;char name[20];
};
struct Stu
{int id;char name[20];
};#include "test.h"
#include "test.h"
#include "test.h"
#include "test.h"
#include "test.h"
int main()
{return 0;
}

为了避免这种情况,所以我们用到了:条件编译,方法如下:

//头文件的改进如下:
//方法1:条件编译
#ifndef __TEST_H__
#define __TEST_H__//也就是说,如果没有__TEST_H__的话,定义一个__TEST_H__,这样后续就有了__TEST_H__,就不会再包含这个头文件了
struct Stu
{int id;char name[20];
};
#endif//方法2:
#pragma once
//这是一个预处理指令,用于告诉编译器该头文件之应该被包含一次,通常推荐放在头文件开头

13.其他预处理指令

#error
#pragma
#line
...
不做介绍,⾃⼰去了解

可以参考《C语言深度解剖》一书进行学习

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

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

相关文章

13. UDP协议与RTP协议

UDP协议 UDP协议比较简单&#xff1a; UDP的长度是固定的&#xff0c;用总长度-UDP长度就是数据长度。 UDP是不保证他的有序性和可靠性的。对于音频和视频是这样是比较好的&#xff0c;因为这段丢了&#xff0c;我们可以从下一段在开始解码。 RTP RTP 协议概述 RTP&#x…

【MySQL】(基础篇六) —— 过滤数据

过滤数据 本文将讲授如何使用SELECT语句的WHERE子句指定搜索条件。 WHERE子句 数据库表一般包含大量的数据&#xff0c;很少需要检索表中所有行。通常只会根据特定操作或需要提取表数据的子集。只检索所需数据需要指定搜索条件&#xff08;search criteria&#xff09;&…

代码随想录算法训练营第36期DAY56

DAY56 套磁很顺利&#xff0c;发现又有书读了&#xff01; 300最长递增子序列 朴素法&#xff0c;这个好想&#xff0c;但是不对&#xff0c;比如 0 1 0 3 2 3 我的算法会找出0 1 3作为答案&#xff0c;而不是0 1 2 3 可以看出&#xff0c;后面的状态依赖于前面的状态&am…

Facebook革新:数字社交的下一个阶段

在数字化时代&#xff0c;社交网络已经成为人们生活中不可或缺的一部分。作为全球最大的社交网络平台之一&#xff0c;Facebook一直在不断创新&#xff0c;引领着数字社交的发展。然而&#xff0c;随着科技的不断进步和社交需求的变化&#xff0c;Facebook正在走向一个新的阶段…

Gitte的使用(Windows/Linux)

Gitte的使用&#xff08;Windows/Linux&#xff09; 一、Windows上使用Gitte1.下载程序2.在Gitte上创建远程仓库3.连接远程仓库4.推送文件到远程仓库 二、Linux上使用Gitte1.第一次从仓库上传1.1生成公钥1.2配置SSH公钥1.3新建一个仓库1.4配置用户名和邮箱在Linux中1.5创建仓库…

python字典应用

""" 字典应用 字典中保存了股票信息&#xff0c;完成下面的操作 1.找出股票价格大于100元的股票并创建一个新的字典 2、找出价格最高和最低的股票对应的股票代码 3.按照股票价格从高到低给股票代码排序 """stocks {AAPL: 191.88,G00G: 1186.96,…

强烈推荐 Setapp 上的 Mac 优质软件

Setapp 一款专为 macOS 设计的软件订阅平台&#xff0c;目前提供高达 240 款精心筛选的高品质应用程序&#xff0c;只需每月 9.9 美元的订阅费&#xff0c;即可畅享所有正版软件的使用权。让使用者无忧享受正版软件的稳定性和安全性&#xff0c;彻底告别盗版软件可能引发的风险…

【C++11】常见的c++11新特性(一)

文章目录 1. C11 简介2. 常见的c11特性3.统一的列表初始化3.1initializer_list 4. decltype与auto4.1decltype与auto的区别 5.nullptr6.右值引用和移动语义6.1左值和右值6.1.1左值的特点6.1.2右值的特点6.1.3右值的进一步分类 6.2左值引用和右值引用以及区别6.2.1左值引用6.2.2…

ELK组件

资源列表 操作系统 IP 主机名 Centos7 192.168.10.51 node1 Centos7 192.168.10.52 node2 部署ELK日志分析系统 时间同步 chronyc sources -v 添加hosts解析 cat >> /etc/hosts << EOF 192.168.10.51 node1 192.168.10.52 node2 EOF 部署Elasticsea…

开源VisualFreebasic中文版,vb7 IDE,VB6升级64位跨平台开发安卓APP,Linux程序

吴涛老矣&#xff0c;社区苦无64位易语言&#xff0c;用注入DLL增强菜单&#xff0c;做成VS一样的界面 终归是治标不治本&#xff0c;一来会报毒&#xff0c;二来闭源20年没更新了 开源的VB7&#xff0c;欢迎易语言的铁粉进群&#xff1a;1032313876 【Freebasic编程语言】编绎…

【设计模式深度剖析】【4】【行为型】【策略模式】

&#x1f448;️上一篇:职责链模式 设计模式-专栏&#x1f448;️ 文章目录 策略模式定义英文原话直译 角色类图策略接口Strategy&#xff1a;具体策略类上下文类Context测试类 策略模式的应用策略模式的优点策略模式的缺点策略模式的使用场景 策略模式 策略模式&#xff08…

Java--递归

1.递归就是A方法调用A方法&#xff0c;也就是调用自己本身 2.利用递归可以用简单的程序来解决一些复杂的问题&#xff0c;它通常把一个大型复杂的问题层层转化为一个与原问题相似的规模较小的问题来求解&#xff0c;递归策略只需少量的程序就可描述出解题过程所需要的多次重复…

Ubuntu 24.04 LTS 安装配置 MySQL Community Server 8.4.0 LTS

1 安装 Apt Repository ​​​​​​​地址MySQL :: Download MySQL APT Repository sudo dpkg -i mysql-apt-config_0.8.30-1_all.deb #安装mysql 8.4 lts sudo apt update sudo apt-get install mysql-server #修改mysql root密码策略 2 查看版本 testtest:~$ mysqld --v…

力扣每日一题 6/10

881.救生艇[中等] 题目&#xff1a; 给定数组 people 。people[i]表示第 i 个人的体重 &#xff0c;船的数量不限&#xff0c;每艘船可以承载的最大重量为 limit。 每艘船最多可同时载两人&#xff0c;但条件是这些人的重量之和最多为 limit。 返回 承载所有人所需的最小船…

LinkedList详解(含数据结构动画演示)

目录 LinkedList详解1、LinkedList的继承体系2、LinkedList的构造函数3、LinkedList的add(E e)方法4、LinkedList的Node节点5、双向链表的概念和Node节点的详细解释6、LinkedList的add(E e)方法梳理7、LinkedList的getXXX方法8、LinkedList的removeXXX方法①、removeFirst()方法…

Tomcat源码解析(八):一个请求的执行流程(附Tomcat整体总结)

Tomcat源码系列文章 Tomcat源码解析(一)&#xff1a;Tomcat整体架构 Tomcat源码解析(二)&#xff1a;Bootstrap和Catalina Tomcat源码解析(三)&#xff1a;LifeCycle生命周期管理 Tomcat源码解析(四)&#xff1a;StandardServer和StandardService Tomcat源码解析(五)&…

吴恩达神经网络学习笔记1

代码解释 并不是全部代码&#xff0c;思路的流程 import numpy as np# 如何判断咖啡豆是烤好了 # 假设此神经网络由2层构成###### 这部分代码只是如何建立2层网络&#xff0c; ###### 并不包含如何加载神经网络中的参数 w 和 b######################## 第1层网络# x 是…

Ruoyi5.x RuoYi-Vue-Plus新建Translation翻译类

若依框架&#xff08;RuoYi&#xff09;中的Translation翻译类主要作用在于实现字段值的转换或翻译功能&#xff0c;以提高数据展示的准确性和友好性。以下是其具体作用的一些关键点&#xff1a; 字段值转换&#xff1a;若依框架在处理数据时&#xff0c;有时需要将某些字段的…

CrawlSpace爬虫部署框架介绍

CrawlSpace爬虫部署框架介绍 全新的爬虫部署框架&#xff0c;为了适应工作的爬虫部署的使用&#xff0c;需要自己开发一个在线编写爬虫及部署爬虫的框架&#xff0c;框架采用的是Django2.2bootstap依赖scrapyd开发的全新通用爬虫在线编辑部署及scrapy项目的部署框架。项目实现的…

读AI未来进行式笔记08自主57

1. 自主57 1.1. 自主57被视为继火药、核57之后的“第三次zhan筝革命” 1.2. 虽然地雷和导弹揭开了早期简单自主57的序幕&#xff0c;但运用了AI技术的真正的自主57才是正片 1.2.1. AI自主57让整个sha戮过程&#xff1a;搜寻目标、进入zhan斗、抹sha生命&#xff0c;完全无须…