《C语言深度剖析》第三章 预处理详解 p2(完结) C语言从入门到入土(进阶篇)(二)

简介: 本章节文章是作者通过观看《C语言深度剖析》等各种资料总结的精华,基础部分省略了不少,是为了让大家能够更加深入了解C语言的魅力!因为为了避免与之前的文章发生赘述,所以就直接讲作者认为的精华部分哈!现在正文开始!

1.1.5.  if defined()多条件判定

16.png

当然也可以这样:

13.png14.png


1.1.6. 条件编译支持嵌套

15.png


1.1.7. #else defined()

17.png


这就可以解决之前说的没有 else if 情况了。

当然如果C CPP 都定义了,就只保留上面的(打印 hello C)。


1.2.1. 为何要有条件编译?

本质认识:条件编译,其实就是编译器根据实际情况,对代码进行裁剪。而这里“实际情况”,取决于运行平台,代码本身的

业务逻辑等。

可以认为有两个好处:

1. 可以只保留当前最需要的代码逻辑,其他去掉。可以减少生成的代码大小

2. 可以写出跨平台的代码,让一个具体的业务,在不同平台编译的时候,可以有同样的表现


1.2.2. 条件编译都在哪些地方用?

举一个例子吧

我们经常听说过,某某版代码是完全版/精简版,某某版代码是商用版/校园版,某某软件是基础版/扩展版等。

其实这些软件在公司内部都是项目,而项目本质是有多个源文件构成的。所以,所谓的不同版本,本质其实就是功能的有无,在技术层面上,公司为了好维护,可以维护多种版本,当然,也可以使用条件编译,你想用哪个版本,就使用哪种条件进行裁剪就行。

著名的Linux内核,功能上,其实也是使用条件编译进行功能裁剪的,来满足不同平台的软件。


2. 文件包含


2.1.1 防止头文件重复包含两种:


一:#pragma once


二:  

18.png

但是为什么写头文件先写这个呢?(如何做的,结果我们是知道的)

第一次 _TEST_H_ 没有被定义,接着 #define 定义,然后后面再定义了,

ifndef 为假不执行,就不会重复定义了。


2.1.2用 “” 包含的头文件

19.png

用“ ” 是现在当前目录文件下查找头文件,找不到再去系统文件目录下查找头文件。

而用 <>是直接在系统文件目录下查找头文件。


PS:前面提到了头文件展开,那到底是什么意思呢?


我们发现我们原来的几行代码,但是看预处理有800多行,简单理解就是把

头文件的内容拷贝到目标文件(也有去注释,条件编译等)。


2.2. #error 预处理

预处理都是在编译期间起效果的,也就是和后面的链接,运行没有关系。这点要注意 另外,很多预处理用的并不多,我们这里挑重要的说一下。

20.png


2.3. #line 预处理

21.png22.png


2.4. #pragma


23.png24.png


他们两个的区别就是一个是提示级别的信息,一个是报错级别的信息。


2.5. # 运算符

25.png

相邻字符串具有自动联接特性。


2.5.1. printf

26.png

注意printf第一个参数就是char*,所以可以直接printf(str),至于后面的 ... 是可变参数列表,后面讲。

27.png

将参数符号s对应的文本内容,转换为对应的字符串。

这里就是一开始s为3.1415926然后转换为#3.1415926然后把3.1415926转换为字符串,再打印出来了。


下面是linux里面的,大家可以看右边是预处理之后的结果。

28.png


2.5.2. #作用:

29.png

我们现在就可以直接把数字12345直接通过我们定义的 TOSTRING 直接转换为字符串!打印出来的就是12345字符串。

这就比我们之前计算而来的方法要便捷得多!


但是!

30.png

这样是不可以的,因为宏是预处理的过程,而开辟空间也是最少到汇编了,所以先一步直接把abc转换为了字符串,打印出来的就是abc。

所以我们要用的话直接写要转的目标就可以了。



2.6. ## 预算符


2.6.1. ##含义

简单说就是把##两边的符号连接起来形成新的符号(如student1就是一个新的符号)。31.png


2.6.2. ##作用:

就比如说我们要打印一个科学记数法的数我们知道底数和他的次方数,就可以用##定义一个新的符号。值得注意的是 ##中写的e ,在编译过程中是有效的!

32.png

33.png

例如:

我们来看看linux预处理后的代码(右边)


34.png35.png


2.6.3. 怎么区分符号和字符串?

其实只要记住字符串是用“”引起的,对于字符像我们写的char* str=“abcd”的str,int a的a,还有我们上面的12345转字符串的12345都是符号!还有字符串是程序内部的内容,编译器不会动他,而符号是可被编译器进行改变的(改为编译器所理解的)。


今天的内容就到这里了哈!!!

要是认为作者有一点帮助你的话!

就来一个点赞加关注吧!!!当然订阅是更是求之不得!

最后的最后谢谢大家的观看!!!

你们的支持是作者写作的最大动力!!!

下期见哈!!!

相关文章
|
3月前
|
编译器 C语言
C语言--预处理详解(1)
【10月更文挑战第3天】
|
3月前
|
编译器 Linux C语言
C语言--预处理详解(3)
【10月更文挑战第3天】
|
2月前
|
C语言
【c语言】你绝对没见过的预处理技巧
本文介绍了C语言中预处理(预编译)的相关知识和指令,包括预定义符号、`#define`定义常量和宏、宏与函数的对比、`#`和`##`操作符、`#undef`撤销宏定义、条件编译以及头文件的包含方式。通过具体示例详细解释了各指令的使用方法和注意事项,帮助读者更好地理解和应用预处理技术。
33 2
|
3月前
|
编译器 Linux C语言
【C语言篇】编译和链接以及预处理介绍(下篇)
【C语言篇】编译和链接以及预处理介绍(下篇)
39 1
【C语言篇】编译和链接以及预处理介绍(下篇)
|
3月前
|
C语言
C语言--预处理详解(2)
【10月更文挑战第3天】
|
3月前
|
编译器 C语言
C语言预处理详解
C语言预处理详解
|
3月前
|
存储 C语言
【C语言篇】编译和链接以及预处理介绍(上篇)2
【C语言篇】编译和链接以及预处理介绍(上篇)
44 0
|
8天前
|
存储 算法 C语言
【C语言程序设计——函数】素数判定(头歌实践教学平台习题)【合集】
本内容介绍了编写一个判断素数的子函数的任务,涵盖循环控制与跳转语句、算术运算符(%)、以及素数的概念。任务要求在主函数中输入整数并输出是否为素数的信息。相关知识包括 `for` 和 `while` 循环、`break` 和 `continue` 语句、取余运算符 `%` 的使用及素数定义、分布规律和应用场景。编程要求根据提示补充代码,测试说明提供了输入输出示例,最后给出通关代码和测试结果。 任务核心:编写判断素数的子函数并在主函数中调用,涉及循环结构和条件判断。
45 23
|
8天前
|
算法 C语言
【C语言程序设计——函数】利用函数求解最大公约数和最小公倍数(头歌实践教学平台习题)【合集】
本文档介绍了如何编写两个子函数,分别求任意两个整数的最大公约数和最小公倍数。内容涵盖循环控制与跳转语句的使用、最大公约数的求法(包括辗转相除法和更相减损术),以及基于最大公约数求最小公倍数的方法。通过示例代码和测试说明,帮助读者理解和实现相关算法。最终提供了完整的通关代码及测试结果,确保编程任务的成功完成。
34 15
|
8天前
|
C语言
【C语言程序设计——函数】亲密数判定(头歌实践教学平台习题)【合集】
本文介绍了通过编程实现打印3000以内的全部亲密数的任务。主要内容包括: 1. **任务描述**:实现函数打印3000以内的全部亲密数。 2. **相关知识**: - 循环控制和跳转语句(for、while循环,break、continue语句)的使用。 - 亲密数的概念及历史背景。 - 判断亲密数的方法:计算数A的因子和存于B,再计算B的因子和存于sum,最后比较sum与A是否相等。 3. **编程要求**:根据提示在指定区域内补充代码。 4. **测试说明**:平台对代码进行测试,预期输出如220和284是一组亲密数。 5. **通关代码**:提供了完整的C语言代码实现
47 24

热门文章

最新文章