linux环境下C程序的编译过程以及makefile的简单使用

devtools/2024/10/20 19:32:24/

        在windows下,很多用来进行编程软件对于写好的文件,点击编译即可生成想要文件。如.exe可执行文件,.hex文件或者.bin文件等等。软件为我们省略了很多事。但是对于linux初学者来说,初次接触linux系统,面对命令行黑框框有点无从下手,也不明白linux下C程序不用可视化编译器如何对程序进行编译,生成可执行文件。

一、C文件的编译过程 

        C语言的编译过程是将源代码转换成计算机能够被理解的机器代码的过程。一般来说主要为以下流程。

C文件->预处理->编译->汇编->连接->可执行文件。

1.1预处理

        首先是预处理阶段,主要对源代码的预处理指令进行处理,如头文件 #include <stdio.h>、宏定义 #define  XX 等等。将#include <xx.h>替换成头文件中的东西,并删除不属于程序的内容,如程序中的注释。预处理完输出的文件被称为pp文件,通常以.i或者.ii结尾。

1.2编译

        编译阶段,通过编译器将预处理后的文件转换为汇编语言代码。在这个过程中,编译器会进行词法分析和语法分析,检查代码的语法和语义,确保代码的正确性。同时,编译器还会生成与平台相关的汇编代码。编译阶段完成后,会生成一个或多个汇编文件,这些文件是用汇编语言编写的,包含了与C语言源代码对应的机器代码指令。

1.3汇编

        汇编阶段,将编译阶段生成的汇编码转换为目标文件,汇编器将汇编代码转换为机器码,并生成一个或者多个目标文件.o或.obj文件。

1.4链接

        链接过程是将汇编阶段生成的目标文件或者多个目标文件以及库文件合并成一个可执行文件。链接器还会进行地址解析和重定位等工作,以确保最终生成的可执行文件中的地址正确无误。链接阶段完成后,会生成最终的可执行文件,这个文件可以直接在操作系统上运行。

二、GCC编译器

        gcc编译器是linux环境中使用的比较多的一个编译器,gcc(GNU Compiler Collection,GNU编译器套件)。

        在linux下可以直接通过命令行查看gcc编译器的使用方法

man gcc

        在命令行界面中直接输入man gcc即可查看使用方法

gcc         xx1        xx2

xx1为选项,即使用gcc进行什么操作 xx1可为以下几种选项

 -c: 只编译不链接为可执行文件,编译器将输入的.c 文件编译为.o 的目标文件。


 -o: <输出文件名>用来指定编译结束以后的输出文件名,如果不使用这个选项的话 GCC 默
认编译出来的可执行文件名字为 a.out。 例 gcc -o hello hello.c  hello为生成的可执行文件的名字,hello.c为需要编译的c文件。


 -g: 添加调试信息,如果要使用调试工具(如 GDB)的话就必须加入此选项,此选项指示编
译的时候生成调试所需的符号信息。


 -O: 对程序进行优化编译,如果使用此选项的话整个源代码在编译、链接的的时候都会进
行优化,这样产生的可执行文件执行效率就高。


-O2: 比-O 更幅度更大的优化,生成的可执行效率更高,但是整个编译过程会很慢。

2.1 编译流程

       GCC 编译器的编译流程是:预处理、编译、汇编和链接。预处理就是展开所有的头文件、
替换程序中的宏、解析条件编译并添加到文件中。编译是将经过预编译处理的代码编译成汇编
代码,也就是我们常说的程序编译。汇编就是将汇编语言文件编译成二进制目标文件。链接就
是将汇编出来的多个二进制目标文件链接在一起,形成最终的可执行文件,链接的时候还会涉
及到静态库和动态库等问题。

2.2 单个文件编译运行

      对于简单的单个的C程序,只需要调用系统的头文件,而不需要调用外部自己编写的c程序,那么只需要使用 gcc -o  xx   xx.c即可。如下所示。

在命令行中输入

vi hello.c

在hello.c中添加一个简单的c程序

  #include <stdio.h>int main(void){printf("hello world\r\n");return 0;}

保存后退出,使用gcc编译

gcc -o hello hello.c

使用ls 即可查看到生成的hello文件

        

运行 ./hello  即可看到命令行输出打印信息

2.3 多个文件编译链接运行

        通过键盘输入两个整形数字,然后计算他们的和并将结果显示在屏幕上,在这个工程中我们有 main.c、 input.c 和 calcu.c 这三个 C 文件和 input.h、 calcu.h 这两个头文件。其中 main.c 是主体, input.c 负责接收从键盘输入的数值, calcu.c 进行任意两个数相加。

在文件夹中创建以下几个文件

 main.c

  1 #include <stdio.h>2 #include "input.h"3 #include "calcu.h"4 5 int main(int argc, char *argv[])6 {7  int a, b, num;8  input_int(&a, &b);9  num = calcu(a, b);10  printf("%d + %d = %d\r\n", a, b, num);11  }

input.c

#include <stdio.h>
#include "input.h"void input_int(int *a, int *b){printf("input two num:");scanf("%d %d", a, b);printf("\r\n");}

calcu.c

#include "calcu.h"
int calcu(int a, int b)
{return (a + b);
}

input.h

#ifndef _INPUT_H
#define _INPUT_Hvoid input_int(int *a, int *b);#endif

calcu.h

#ifndef _CALCU_H
#define _CALCU_Hint calcu(int a, int b);#endif

输入命令行 

gcc main.c calcu.c input.c -o main

对 main.c calcu.c input.c 进行编译并链接成 main 可执行文件

运行结果

三、 makefile引入使用

3.1 实例

对于2.3中的例子,只需要一行代码即可解决编译,看起来还是挺简单的。但是在实际项目中,有很多.c和.h文件,如果每次都这样编译链接的话可就太麻烦了。因此引入makefile简化这个编译流程。

在当前文件中创建一个 Makefile文件

vi Makefile

在代码中添加

  1 main: main.o input.o calcu.o2         gcc -o main main.o input.o calcu.o3 main.o: main.c4         gcc -c main.c5 input.o: input.c6         gcc -c input.c7 calcu.o: calcu.c8         gcc -c calcu.c9 10 clean:11         rm *.o12         rm main

写好makefile后,使用make 即可对多个文件进行编译、链接

make

可以看到编译器单独对 main.c input.c calcu.c 进行了编译,最后链接生成了main.o可执行文件

对于多个C文件来说,方便了很多。

3.2 makefile解释

  1 main: main.o input.o calcu.o2         gcc -o main main.o input.o calcu.o3 main.o: main.c4         gcc -c main.c5 input.o: input.c6         gcc -c input.c7 calcu.o: calcu.c8         gcc -c calcu.c9 10 clean:11         rm *.o12         rm main
   main: main.o input.o calcu.ogcc -o main main.o input.o calcu.o

        main 是最后需要生成的目标文件,而main.o input.o calcu.o是生成main所需要的依赖文件
如果要更新目标 main,就必须先更新它的所有依赖文件,如果依赖文件中的任何一个有更新,那么目标也必须更新,“更新”就是执行一遍规则中的命令列表。 并通过gcc 命令将main.o input.o calcu.o链接成可执行文件main。

   main.o: main.cgcc -c main.c

如上述代码差不多,main.o为目标文件,main.c为依赖文件,通过执行gcc -c main.c生成目标文件。input.o calcu.o跟这个一样就不过多阐述了。

  clean:rm *.orm main

        当需要清除生成的目标文件时,可通过命令行控制,删除以.o结尾的目标文件 ,删除可执行文件 main。

可以看到,输入make clean命令后,以上文件皆被删除了。


http://www.ppmy.cn/devtools/127362.html

相关文章

什么是同步锁?

同步锁&#xff08;Synchronized Lock&#xff09;是多线程编程中的一个关键概念&#xff0c;用来解决多个线程同时访问共享资源时可能引发的数据不一致问题。同步锁通过限制多个线程并发执行关键代码的能力&#xff0c;确保在同一时刻只有一个线程能够访问临界区&#xff08;即…

【DBA Part03】国产Linux上Oracle RAC安装-升级-ADG-迁移

本阶段内容如下&#xff1a; 01.国产统信UOS-Oracle19c安装配置 02.国产龙蜥AnolisOS-Oracle19c RAC集群安装配置 03.Linux-Oracle11gR2数据库升级到Oracle19C 04.Linux-Oracle11gR2 RAC数据库升级到Oracle19c RAC 05.Linux-Oracle19c ADG容灾配置(111级联) 06.Oracle XT…

目标检测系统中需要【重新训练模型】说明

上百种【基于YOLOv8/v10/v11的目标检测系统】目录&#xff08;pythonpyside6界面系统源码可训练的数据集也完成的训练模型&#xff09;-CSDN博客 目标检测系统操作说明【用户使用指南】&#xff08;pythonpyside6界面系统源码可训练的数据集也完成的训练模型&#xff09;-CSDN…

C语言:函数指针与指针函数的区别*

文章目录 一、函数指针定义语法 二、指针函数定义语法用途 三、函数指针与指针函数的区别本质不同&#xff1a;声明方式&#xff1a; 四、结论 C语言&#xff1a;函数指针与指针函数的区别 在C语言这个充满灵活性和强大表达力的编程世界中&#xff0c;函数指针和指针函数是两个…

PDT 数据集:首个基于无人机的高精密度树木病虫害目标检测数据集

2024-09-24&#xff0c;由中国山东计算机科学中心、北京大学等机构联合创建了Pests and Diseases Tree&#xff08;PDT&#xff09;数据集&#xff0c;目的解决农业领域中病虫害检测模型开发中专业数据集缺失的问题。通过集成公共数据和网络数据&#xff0c;进一步推出了Common…

Kafka 为什么要抛弃 Zookeeper?

嗨&#xff0c;你好&#xff0c;我是猿java 在很长一段时间里&#xff0c;ZooKeeper都是 Kafka的标配&#xff0c;现如今&#xff0c;Kafka官方已经在慢慢去除ZooKeeper&#xff0c;Kafka 为什么要抛弃 Zookeeper&#xff1f;这篇文章我们来聊聊其中的缘由。 Kafka 和 ZooKee…

前端开发攻略---取消已经发出但是还未响应的网络请求

目录 注意&#xff1a; 1、Axios实现 2、Fetch实现 3、XHR实现 注意&#xff1a; 当请求被取消时&#xff0c;只会本地停止处理此次请求&#xff0c;服务器仍然可能已经接收到了并处理了该请求。开发时应当及时和后端进行友好沟通。 1、Axios实现 <!DOCTYPE html> &…

【VUE】Vue中的 keep-alive 组件

Vue2中的keep-alive组件主要用来缓存组件实例,以便在切换时保留其状态。这样能够提高应用程序的性能,避免了在多个页面之间频繁地创建和销毁组件实例。常用于:多表单切换,对表单内数据进行保存。 使用keep-alive组件时需要注意以下几点: keep-alive组件只能用于包含动态组…