• 【C语言】进阶——程序编译


     

    目录

    一:🔒程序环境

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

    💡1.1翻译环境  

    预编译阶段:

    编译阶段:

    汇编阶段:

    链接阶段:

    💡1.2运行环境 

     二:🔒预处理详解

    💡2.1预处理符号

    💡2.2#define 

    #define定义标识符

    #define定义宏

    #define替换规则 

     🔒#和## 

    💡#的作用

    💡##的作用

    带副作用的宏参数 

    三:宏与函数的对比 

    💡命名约定 

     四:🔒条件编译

    五:🔒文件包含

    💡头文件的包含方式 

    💡避免头文件被重复引用 


     

     

    一:🔒程序环境

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

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

            1.翻译环境,在这个环境下源代码被转换成可执行的机器指令

            2.执行环境,用于实际代码运行

     

    💡1.1翻译环境  

    1.组成一个程序的每个源文件通过编译过程分别转换成目标代码(object code)。
    2.每个目标文件由链接器(linker)捆绑在一起,形成一个单一而完整的可执行程序。
    3.链接器同时也会引入标准C函数库中任何被该程序所用到的函数,而且它可以搜索程序员个人的程序库,将其需要的函数也链接到程序中。 

     

    预编译阶段:

    1.头文件包含

    #include 预处理指令

    2.define定义的符号替换

    #define 预处理指令

    3.注释删除

    以上这些都是文本操作

    编译阶段:

    把c语言代码翻译成了汇编代码

    1、语法分析

    2、词法分析

    3、语义分析

    4、符号汇总 

    汇编阶段:

    汇编指令翻译成了二进制的指令

    形成符号表,这样就能够找到源文件外部的符号(只能汇总全局符号)

    链接阶段:

    1、合并段表

    2、符号表的合并和重定位

    💡1.2运行环境 

    程序执行的过程:

    1. 程序必须载入内存中。在有操作系统的环境中:一般这个由操作系统完成。在独立的环境中,程序 的载入必须由手工安排,也可能是通过可执行代码置入只读内存来完成。

    2. 程序的执行便开始。接着便调用main函数。

    3. 开始执行程序代码。这个时候程序将使用一个运行时堆栈(stack),存储函数的局部变量和返回地址。程序同时也可以使用静态(static)内存,存储于静态内存中的变量在程序的整个执行过程一直保留他们的值。

    4. 终止程序。正常终止main函数;也有可能是意外终止。

     二:🔒预处理详解

    💡2.1预处理符号

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

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

    1. int main()
    2. {
    3. printf("file:%s line:%d\n", __FILE__, __LINE__);
    4. return 0;
    5. }

    💡2.2#define 

    #define定义标识符

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

    在宏定义时,最好不要加分号 ( ; )

    因为宏定义标识符,并不会进行计算,在编译阶段进行的是内容替换 

    1. #define MAX 100;
    2. int main()
    3. {
    4. int max = 0;
    5. if (1)
    6. max = MAX; //error
    7. else
    8. max = 0;
    9. return 0;
    10. }

    这里宏定义会直接替换,将 100; 替换到 MAX位置

    1. if (1)
    2. max = 100;
    3. ;
    4. else
    5. max = 0;

    #define定义宏

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

    1. #define name(parament-list) stuff
    2. //name表示名字
    3. //parament-list 是以逗号隔开的参数
    4. //宏的具体内容
    5. 例子
    6. //#define max(a,b) a+b

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

    同样也要注意因为宏是直接进行文本替换,然后才在程序中发生计算,所以如果不按照标准规定写宏,可能会产生bug 

    1. #define SQUARE(x) x*x
    2. int c=5SQUARE(5+1);
    3. //我们预期这里的内容是36,但是最终结果是11,是因为实际计算的是
    4. //5+1*5+1==11

    所以在写的时候我们应该尽可能带上括号,防止因为优先级的问题出现bug

    #define SQUARE(x) ((x)*(x))

    #define替换规则 

    在程序中扩展 #define 定义符号和 宏 时,需要涉及几个步骤。

    1. 在调用宏时,首先对参数进行检查,看看是否包含任何由#define定义的符号。如果是,它们首先 被替换。

    2. 替换文本随后被插入到程序中原来文本的位置。对于宏,参数名被他们的值替换。

    3. 最后,再次对结果文件进行扫描,看看它是否包含任何由#define定义的符号。如果是,就重复上 述处理过程。.

    注意:

    宏参数和#define 定义中可以出现其他#define定义的变量。但是对于宏,不能出现递归
    当预处理器搜索#define定义的符号的时候,字符串常量的内容并不被搜索。

     🔒#和## 

    💡#的作用

     如何把参数插入到字符串中?

    1. int main()
    2. {
    3. printf("who say!!!\n");
    4. printf("who" " say!!!\n");
    5. return 0;
    6. }

     

    字符串是具备自动连接的特点。

     而define定义的符号,在"X" 里面不会被识别,我们可以用 #X 解决此问题

    1. #define PRINT(n, format) printf("the value of "#n" is "format"\n", n)
    2. int main()
    3. {
    4. int a = 10;
    5. PRINT(a, "%d");
    6. //printf("the value of ""a"" is ""%d""\n", a);
    7. //printf("the value of a is %d\n", a);
    8. int b = 20;
    9. PRINT(b, "%d");
    10. return 0;
    11. }

    💡##的作用

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

    1. #define CAT(v, n) v##n
    2. int main()
    3. {
    4. int value10 = 100;
    5. printf("%d\n", CAT(value, 10));
    6. printf("%d\n", value10);
    7. return 0;
    8. }

    注:
    这样的连接必须产生一个合法的标识符。否则其结果就是未定义的。 

    带副作用的宏参数 

    简单来讲就是宏在执行的过程中,参数自身的值会发生变化,这个就叫做带副作用的宏的参数

    1. #define MAX(a,b) ((a)>(b)?(a):(b))
    2. int main()
    3. {
    4. int x=5,y=8;
    5. int c=MAX(x++,y++);
    6. printf("%d ",c);
    7. return 0;
    8. }
    9. //这里输出的是多少,9嘛?
    10. //首先带入 ((5++>(8++)?(a++):(b++))
    11. 首先是进行ab大小的比较,在这里比较之后,
    12. //a跟b跟别变成了69,然后执行后面的b++,最终的结果应该a=6 c=9 b=10

    要避免写出这样的代码,宏是无法调试的

    三:宏与函数的对比 

     宏通常被应用于执行简单的运算。
    比如在两个数中找出较大的一个。

    宏定义和函数的比较 

    属性#define宏定义函数
    代码长度每次使用时,宏代码都会被插入到程序中。除了非常 小的宏之外,程序的长度会大幅度增长函数代码只出现于一个地方;每 次使用这个函数时,都调用那个、地方的同一份代码
    执行速度  相对更快(简单的程序)存在函数的调用和返回的额外开 销,所以相对慢一些

    操作符

    优先级

    宏参数的求值是在所有周围表达式的上下文环境里, 除非加上括号,否则邻近操作符的优先级可能会产生 不可预料的后果,所以建议宏在书写的时候多些括 号。函数参数只在函数调用的时候求 值一次,它的结果值传递给函 数。表达式的求值结果更容易预 测。
    带 有 副 作 用 的 参 数惨数可能被替换到宏体中的多个位置,所以带有副作 用的参数求值可能会产生不可预料的结果。函数参数只在传参的时候求值一 次,结果更容易控制。
    参数类型 宏的参数与类型无关,只要对参数的操作是合法的, 它就可以使用于任何参数类型。函数的参数是与类型有关的,如 果参数的类型不同,就需要不同 的函数,即使他们执行的任务是 不同的。
    调试无法调试函数可以逐语句调试的
    递归无法递归可以递归

    1.用于调用函数和从函数返回的代码可能比实际执行这个小型计算工作所需要的时间更多。
    所以宏比函数在程序的规模和速度方面更胜一筹。(相对简单定义)
    2.更为重要的是函数的参数必须声明为特定的类型。
    所以函数只能在类型合适的表达式上使用。

    反之这个宏怎可以适用于整形、长整型、浮点型等可以
    用于来比较的类型。
    宏是类型无关的。

    💡命名约定 

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

    把宏名全部大写
    函数名不要全部大写(一般单词首字母大写)

     四:🔒条件编译

    条件编译顾名思义就是满足条件才进行读取

    1. 1.
    2. #if 常量表达式
    3. //...
    4. #endif
    5. //常量表达式由预处理器求值。
    6. 如:
    7. #define __DEBUG__ 1
    8. #if __DEBUG__
    9. //..
    10. #endif
    11. 2.多个分支的条件编译
    12. #if 常量表达式
    13. //...
    14. #elif 常量表达式
    15. //...
    16. #else
    17. //...
    18. #endif
    19. 3.判断是否被定义
    20. #if defined(symbol)
    21. #ifdef symbol
    22. #if !defined(symbol)
    23. #ifndef symbol
    24. 4.嵌套指令
    25. #if defined(OS_UNIX)
    26. #ifdef OPTION1
    27. unix_version_option1();
    28. #endif
    29. #ifdef OPTION2
    30. unix_version_option2();
    31. #endif
    32. #elif defined(OS_MSDOS)
    33. #ifdef OPTION2
    34. msdos_version_option2();
    35. #endif
    36. #endif

    五:🔒文件包含

    #include 指令可以使另外一个文件被编译。就像它实际出现于 #include 指令的地方 一样。

    这种替换的方式很简单:

    预处理器先删除这条指令,并用包含文件的内容替换。

    这样一个源文件被包含10次,那就实际被编译10次。

    💡头文件的包含方式 

    本地文件

    #include"test.h"

    先从源文件所在目录进行查找头文件,然后再到标准函数库头文件所在目录下查找

    #include

    直接从标准函数库头文件所在目录下查找

    总的来说就是""的引用方式查找范围更广

    但是!!!

    本地文件还是按照 #include"test.h''  的方式

    库文件按照 #include的方式  

    否则:查找的效率就低些,当然这样也不容易区分是库文件还是本地文件了。

    💡避免头文件被重复引用 

    防止多次头文件频繁引用

    1. #ifndef __TEST_H__
    2. #define __TEST_H__
    3. //头文件的内容
    4. #endif //__TEST_H__

    或者

    #pragma once
    

    以上便是我对【C语言】程序编译的介绍,文中不足之处,还望得到指点得以改善。感谢!!! 

     

  • 相关阅读:
    [附源码]计算机毕业设计springboot疫情网课管理系统
    phpstorm不提示$this->request,不提示Controller父类的方法
    递归神经网络(RNN)在AI去衣技术中的深度应用
    Spring及Spring boot 第四章-第二节 Spring声明式事务管理 @Transactional
    Vue组件化编程详解
    MaxCompute字符串分割函数-SPLIT_PART
    算法通关村第十八关——回溯
    3 网络协议入门
    ARM64汇编基础
    408王道计算机网络强化——数据链路层
  • 原文地址:https://blog.csdn.net/m0_67367079/article/details/133962746