c语言学习第三十五课——程序环境与预处理

简介: c语言学习第三十五课——程序环境与预处理

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

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

第一种环境是翻译环境:在这个环境中源代码被转换为可执行的机器指令。

第二中是执行环境:用于执行实际的代码。

1f0f8f2554474055b707f8f8e0fc7dab.png

我们用图来认识到一个项目是如何编译运行的:

源文件单独通过编译器生成一个目标文件(.obj文件),在利用链接器将众多的可执行程序链接起来


542af4cdafdd4e26a0d6ed6a09db5030.png每一个源文件单独经过编译器编译生成对应的目标文件,我们可在源文件下找到这些目标文件。

在vs编译器上:


9eab1a269cab4ecc96bbc0956e57a5b5.png

编译器是cl.exe文件,链接器是link.exe文件 。

现在详解一下编译环境下的操作:

f90706aae0c4479ea77c82ba9166a6de.png

在进入编译时的

编译分三个阶段

预编译   编译  汇编

gcc test.c   预处理中1.头文件的包含 2.注释的删除及define定义的符号的替换/gcc.test.i -S编译过程中会生成一个.s文件,把c语言代码翻译成汇编代码(通过语法分析,词法分析,语义分析,符号汇总等)参考《编译原理》

汇编  gcc.test -C 该过程会生成一个test.o的文件,在gcc中就是目标文件-一个二进制文件,所以会变就是把汇编代码转换为二进制指令,形成符号表。

预处理阶段会做三个事:                  编译做的事:                    汇编做的事:

这里的编译,感兴趣可以去看《程序员的自我修养》,里面会详细讲到如何翻译成汇编代码。

这里的汇编,会形成符号表。

之后在进行链接:                  

链接  链接所有的.o的可执行文件,包括库形成可执行程序  1.合并段表 2.符号表的合并与重定位

符号(一般是全局变量的与函数的名称)。将这些符号汇总到一起,形成符号表。

因为文本文件的格式是elf格式

合并段表,把对应的段的数据合并到一起

在形成符号表时分模块,static修饰外部链接属性为内部连接属性,即他们符号表合并到一起。

与链接库链接生成可执行文件。

整个过程可理解为就是将高级语言转化位低级语言给计算机识别:

bafaa2e2bbe9481f8026145219cf2061.png

在程序执行的过程中:

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

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

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

4.终止程序,正常种植main函数,亦有可能是意外终止。

预处理详解

1.预定义符号

__LINE__  //文件当前行号

__FUNCTION__//

__FILE__          //进行的源文件

__DATA__        //文件被编译的日期

 __FUNCDNAME__//

__TIME__             //文件被编译的时间

等都是可以展示该文件的属性的指令

#include<stdio.h>
int main()
{
    int i;
    int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
    for (i = 0; i < 10; i++)
    {
        printf("%d------%s,%s %s line=%d\n", arr[i], __FILE__, __DATE__, __TIME__, __LINE__);
    }
    return 0;
}

59e401bbad75464698985330b810ae4f.png

#define

定义一个标识符 完成替换.

这里   \为续行符  在代码后 \之后可以换行,表示还是在一行上。

//在有些计算机语言没有break语句,但在c语言里是必须要有break语句来结束case语句。
//有人就会定义宏来实现不去写break语句
#define CASE  break;case
int main()
{
   int n=0;
   scanf("%d",&n);
   switch(n)
 {
   case 1:
        //....
   CASE 2:
        //.....
   CASE 3:
        //.....
     break;
 }
   return 0;
}

#define 定义宏 替换参数

实现一个数的平方

#define POW(x) x*x
int main()
{
printf("%d", POW(8));
return 0;
}

预处理的操作符#

define print_format(num,format)    printf("the value of "#num" is "format,num)

#的作用是不是替换内容,而是变成对应的字符串

int main()
{
  int a = 10;
  printf("the value of a  is %d", a);
  int b = 20;
  printf("the value of b  is %d", b);
  //我们可以这样写
  print_format(a, "%d\n");
}

预处理的操作符##

可以把位于它两边的符号合成一个符号

int Class110 = 2023;
#define CAT(x,y)  x##y
int main()
{
  printf("%d\n", CAT(Class, 110));//2023  前提是合成的符号具有意义
  return 0;
}

带副作用的宏参数

#define MAX(x,y)    ((x)>(y)?(x):(y))
int main()
{
  //这里想求最大值
  int a = 3;
  int b = 5;
  int c = MAX(3, 5);
  //int c=((3++)>(5++)?(3++):(5++))
  //这里反而不能实现最大值的求解
  printf("%d\n", c);//6
  printf("%d\n", a);//4
  printf("%d\n", b);//7
  return 0;
}

我们从中可以发现。在定义宏参与运算时,我们最好加上括号,防止因为运算优先级而出现问题,因为宏的作用仅仅是替换。

宏和函数对比

1.调用函数的从函数返回的代码比实际执行这个计算需要的时间多。考虑函数栈帧,调用前到运行结束都是需要花费时间的而宏只有主要运算,时间更短,效率更高。

2.宏无类型,函数传参还需定义类型宏的替换可以是类型

如 #define MALLOC(num,type)  (type*)malloc(num,sizoef(type))

3.但是宏只能实现简单的效果,若一个程序实现复杂,宏需多次定义

4.宏无法调试,与类型无关,当然也不够严谨

5.宏会带来运算符优先级的问题

命名约定

宏名都是大写

函数一般部分字母大写

#undef  

取消符号的宏定义

#define MAX 100
#undef MAX;
int main()
{
  int a = 0;
  int b = a + MAX;
  printf("%d", b);
  return 0;
}

这里的MAX未被定义

条件编译

#if   #endif   #if define  #if !define

满足条件可以进入编译 不满足不进入编译
#if (条件)//条件一般为常量表达式
.....
#endif
//多分支
#if  (条件)
....
#elif(条件)
.........
#endif
//判断是否被定义
#if defined (定义的变量)
......
#endif
//如果定义了MAX,语句参与编译
//如果没定义,语句参与编译
#if !defined(定义的变量)
...
#endif
#ifndef MAX
....
#endif

头文件包含

头文件分两种:

库文件#include<stdio.h>  

当前文件#include"file name"

对于两者的区别:

" "先在文件路径查找之后才在库文件查找。

<>包含的头文件会直接去库文件里调用。

我们一般不去将库函数的头文件用“”来调用,这样反而效率会变低。

若头文件被重复包含,里面的内容也被重复的调用进来

可以定义#pragma once  #define __TEST.H__,(头文件)这样重复包含也只调用一次

其他还有许多预处理指令,详见高质量c++/c编程


相关文章
|
2月前
|
存储 算法 C语言
"揭秘C语言中的王者之树——红黑树:一场数据结构与算法的华丽舞蹈,让你的程序效率飙升,直击性能巅峰!"
【8月更文挑战第20天】红黑树是自平衡二叉查找树,通过旋转和重着色保持平衡,确保高效执行插入、删除和查找操作,时间复杂度为O(log n)。本文介绍红黑树的基本属性、存储结构及其C语言实现。红黑树遵循五项基本规则以保持平衡状态。在C语言中,节点包含数据、颜色、父节点和子节点指针。文章提供了一个示例代码框架,用于创建节点、插入节点并执行必要的修复操作以维护红黑树的特性。
68 1
|
2月前
|
NoSQL 编译器 程序员
【C语言】揭秘GCC:从平凡到卓越的编译艺术,一场代码与效率的激情碰撞,探索那些不为人知的秘密武器,让你的程序瞬间提速百倍!
【8月更文挑战第20天】GCC,GNU Compiler Collection,是GNU项目中的开源编译器集合,支持C、C++等多种语言。作为C语言程序员的重要工具,GCC具备跨平台性、高度可配置性及丰富的优化选项等特点。通过简单示例,如编译“Hello, GCC!”程序 (`gcc -o hello hello.c`),展示了GCC的基础用法及不同优化级别(`-O0`, `-O1`, `-O3`)对性能的影响。GCC还支持生成调试信息(`-g`),便于使用GDB等工具进行调试。尽管有如Microsoft Visual C++、Clang等竞品,GCC仍因其灵活性和强大的功能被广泛采用。
95 1
|
2月前
|
编译器 C语言 计算机视觉
C语言实现的图像处理程序
C语言实现的图像处理程序
78 0
|
28天前
|
存储 编译器 程序员
C语言程序的基本结构
C语言程序的基本结构包括:1)预处理指令,如 `#include` 和 `#define`;2)主函数 `main()`,程序从这里开始执行;3)函数声明与定义,执行特定任务的代码块;4)变量声明与初始化,用于存储数据;5)语句和表达式,构成程序基本执行单位;6)注释,解释代码功能。示例代码展示了这些组成部分的应用。
41 10
|
1月前
|
Shell Linux API
C语言在linux环境下执行终端命令
本文介绍了在Linux环境下使用C语言执行终端命令的方法。首先,文章描述了`system()`函数,其可以直接执行shell命令并返回结果。接着介绍了更强大的`popen()`函数,它允许程序与命令行命令交互,并详细说明了如何使用此函数及其配套的`pclose()`函数。此外,还讲解了`fork()`和`exec`系列函数,前者创建新进程,后者替换当前进程执行文件。最后,对比了`system()`与`exec`系列函数的区别,并针对不同场景推荐了合适的函数选择。
|
2月前
|
存储 自然语言处理 程序员
【C语言】文件的编译链接和预处理
【C语言】文件的编译链接和预处理
|
2月前
|
程序员 编译器 C语言
C语言中的预处理指令及其实际应用
C语言中的预处理指令及其实际应用
62 0
|
2月前
|
C语言 索引
C语言编译环境中的 调试功能及常见错误提示
这篇文章介绍了C语言编译环境中的调试功能,包括快捷键操作、块操作、查找替换等,并详细分析了编译中常见的错误类型及其解决方法,同时提供了常见错误信息的索引供参考。
|
C# C语言 C++
从头开始学习c语言
以前的时候学习C语言时候认为C语言不过是一个学习的工具,学习一些理论知识就达到目的了,谁会用这么傻的语言啊,连个界面也没有,不像vb一下子就做出一个窗体来,放上几个按钮就可以了 后来学习C++的时候,认为C++与C是一种完全不同的语言,两者基本没有什么兼容性,当时在学校里看书的时候,一直比较纳闷为会c++的书上会写C++/C语言教程,到了现在才明白,我去C++与C本来就是一体是一脉相承的
1045 0
|
23天前
|
存储 Serverless C语言
【C语言基础考研向】11 gets函数与puts函数及str系列字符串操作函数
本文介绍了C语言中的`gets`和`puts`函数,`gets`用于从标准输入读取字符串直至换行符,并自动添加字符串结束标志`\0`。`puts`则用于向标准输出打印字符串并自动换行。此外,文章还详细讲解了`str`系列字符串操作函数,包括统计字符串长度的`strlen`、复制字符串的`strcpy`、比较字符串的`strcmp`以及拼接字符串的`strcat`。通过示例代码展示了这些函数的具体应用及注意事项。
下一篇
无影云桌面