深圳幻海软件技术有限公司 欢迎您!

【C语言】程序环境和预处理|预处理详解|定义宏(上)

2023-03-14

主页:114514的代码大冒险qq:2188956112(欢迎小伙伴呀hi✿(。◕ᴗ◕。)✿ )Gitee:庄嘉豪(zhuang-jiahaoxxx)-Gitee.com文章目录目录文章目录前言一、程序的翻译环境和执行环境二、详解编译和链接1.翻译环境2.编译本身可分为几个阶段3.运行环境

主页:114514的代码大冒险

qq:2188956112(欢迎小伙伴呀hi✿(。◕ᴗ◕。)✿ )

Gitee:庄嘉豪 (zhuang-jiahaoxxx) - Gitee.com

文章目录

目录

文章目录

前言

一、程序的翻译环境和执行环境

二、详解编译和链接

1.翻译环境

2.编译本身可分为几个阶段

3.运行环境

二,预处理详解

1.预定义符号

2.#define

2.1 #define 定义标识符

2.2#define 定义宏

2.3#define 替换规则

 2.4 #和##

总结



前言

本文尽我可能地使用了足够通俗的语言

希望能够为你带来一些帮助


一、程序的翻译环境和执行环境

在ANSI C的任何一种实现中,存在两个不同的环境

·翻译环境,在这个环境中源代码被转换为可执行的机器指令

·执行环境,它用于实际执行代码

二、详解编译和链接

1.翻译环境

 源文件经编译器处理,转化成后缀为obj(后缀名在不同系统会有所差异)的目标文件

每个源文件在这一过程中都是独立的

所有的源文件经过整合与链接库中的库函数通过链接器生成可执行程序

2.编译本身可分为几个阶段

计算机只能执行二进制指令

test.c-------->test.exe(可执行程序)

我们来看看这个过程具体发生了什么

环境由集成开发环境(IDE)VS2022提供

gcc编译器中的相关指令:
1. 预处理选项 gcc - E test.c - o test.i
预处理完成之后就停下来,预处理之后产生的结果都放在 test.i 文件中。
2. 编译选项gcc -S test.c
3. 汇编gcc -c test.c

3.运行环境

程序执行的过程:
1. 程序必须载入内存中。在有操作系统的环境中:一般这个由操作系统完成。在独立的环境中,程序的载入必须由手工安排,也可能是通过可执行代码置入只读内存来完成。
2. 程序的执行便开始。接着便调用 main 函数。
3. 开始执行程序代码。这个时候程序将使用一个运行时堆栈( stack),存储函数的局部变量和返回 地址。程序同时也可以使用静态( static )内存,存储于静态内存中的变量在程序的整个执行过程 一直保留他们的值。
4. 终止程序。正常终止 main 函数;也有可能是意外终止。
我们这里提一下符号表
简单来讲:

 

有地址的对象只能是全局变量 和函数之类的

不能是临时变量

通过上图的地址的整合,于是形成了一张表格

这个表格就是符号表

瞎编的地址:

然后在链接阶段把无用的变量去掉

二,预处理详解

1.预定义符号

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

 __STDC__     IDE : vs2022未遵循ANSI C,所以无法正常打印

这些预定义符号都是语言内置的

2.#define

2.1 #define 定义标识符

举例:

  1. #define MAX 1000
  2. #define reg register          //为 register这个关键字,创建一个简短的名字
  3. #define do_forever for(;;)     //用更形象的符号来替换一种实现
  4. #define CASE break;case        //在写case语句的时候自动把 break写上。
  5. // 如果定义的 stuff过长,可以分成几行写,除了最后一行外,每行的后面都加一个反斜杠(续行符)。#define DEBUG_PRINT printf("file:%s\tline:%d\t \
  6.                          date:%s\ttime:%s\n" ,\
  7.                          __FILE__,__LINE__ ,       \
  8.                          __DATE__,__TIME__ )  

 [提问]  那么define定义标识符的时候,要不要最后加上; ?

 

测试代码:

  1. #define MAX 1000;
  2. //#define MAX 1000
  3. int main()
  4. {
  5. int m = 0;
  6. if (m >= 0)
  7. m = MAX;
  8. else
  9. m = -1;
  10. printf("%d\n", m);
  11. return 0;
  12. }

 说到定义标识符

我这里忽然想到了经常被用来使用在C语言Switch语句中的定义形式

#define CASE break;case

因为在别的语言中的,大多是不需要在每一个case结束时添加break;

所以为了避免出错

出现了上图所示的定义

定义前

  1. int main()
  2. {
  3. int n = 0;
  4. switch (n)
  5. {
  6. case 1:
  7. break;
  8. case 2:
  9. break;
  10. case 3:
  11. break;
  12. case 4:
  13. break;
  14. case 5:
  15. break;
  16. }
  17. }

定义之后:

  1. int main()
  2. {
  3. int n = 0;
  4. switch (n)
  5. {
  6. case 1:
  7. CASE 2 :
  8. CASE 3 :
  9. CASE 4 :
  10. }
  11. }

2.2#define 定义宏

  1. #define SQUARE(X) X*X
  2. //宏是替换
  3. int main()
  4. {
  5. printf("%d\n", SQUARE(5));
  6. printf("%lf\n", SQUARE(5.0));
  7. printf("%d\n", SQUARE(5+1));//?
  8. return 0;
  9. }

前两个printf打印结果很显然

那么对于最后一个呢?

它的结果是“36”吗?

emmmm,这值得考虑

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

很显然,宏是一种替换,那么

最后的那个printf对应的应该是 “5+1*5+1”

结果应当是“11”

怎么办?出现bug了!

答:加个括号控制一下优先级

#define SQUARE(X) (X)*(X)

嗯,这样就可以完全解决这个bug了

  1. #define DOUBLE(X) (X)+(X)
  2. int main()
  3. {
  4. printf("%d\n", DOUBLE(3));
  5. printf("%d\n",10*DOUBLE(3));
  6. return 0;
  7. }

那么这个宏定义是不是完美的呢?

显然不是啊,后一个printf对应的内容替换之后就成了“10*(3)+(3)”

这并不是我们想要的效果

于是,我们又重新考虑(依然是优先级的问题)

#define DOUBLE(X) ((X)+(X))

嗯,这个就解决了bug

所以在宏定义时,不要吝啬括号,大胆用就是

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

2.3#define 替换规则

 2.4 #和##

前提知识:

 引出:

 这里要写两个printf,太过于繁琐

所以我们希望可以实现一个宏,让它来完成这种效果

 这样显然是不太可行的

于是就有了今天的主角“#”

作用:

把一个宏参数变成对应的字符串

 于是乎:

#define PRINT(x) printf("the value of "#x" is %d\n",x)

比如说传参“a”,经过“#”的作用,就成了“printf("the value of""a"" is %d\n",x)”

也就是三个紧挨着的字符串,这个就可以完美呈现我们想要的效果

---------------------------------------------------------------------------------------------------------------------------------

  1. #define CAT(x,y) x##y
  2. int main()
  3. {
  4. int Class109 = 2023;
  5. printf("%d\n", CAT(Class, 109));
  6. return 0;
  7. }

 由此可得“##”的作用

## 可以把位于它两边的符号合成一个符号。
它允许宏定义从分离的文本片段创建标识符。

总结

那么,这就是本次的全部内容了,下半篇会在隔天之后补上,敬请期待

希望我的文章能够对你有所帮助

文章知识点与官方知识档案匹配,可进一步学习相关知识
算法技能树首页概览40346 人正在系统学习中