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

server/2024/10/20 19:33:25/

        在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/server/133423.html

相关文章

【树莓派 5B】Python 版本切换

【树莓派 5B】Python 版本切换 前言整体思路具体步骤常见问题Python 无法建立与 Python3 的软连接 前言 本文基于树莓派5B 32-bit 树莓派OS&#xff0c;以 Python-3.11.2 降级到 3.9.2 为例&#xff0c;总结了在树莓派上切换 Python 版本的步骤&#xff0c;帮助大家轻松完成 P…

【JS】双指针法获得满足三数之和且不重复的三元组

思路 排序&#xff1a;首先对数组进行排序&#xff0c;这样可以保证在遍历过程中&#xff0c;相同的元素会相邻&#xff0c;同时也方便使用双指针法。 避免重复&#xff1a;在遍历过程中&#xff0c;如果当前元素与前一个元素相同&#xff0c;则跳过当前元素&#xff0c;以避免…

Cloudlog delete_oqrs_line 未授权SQL注入漏洞复现

0x01 产品简介 Cloudlog 是一个自托管的 PHP 应用程序,可让您在任何地方记录您的业余无线电联系人。使用PHP和MySQL构建的基于Web的业余无线电记录应用程序支持从HF到微波的一般站记录任务 0x02 漏洞概述 Cloudlog delete_oqrs_line 接口存在未授权SQL注入漏洞,未经身份验…

nacos的使用

nacos的使用 本专栏的上一篇文章已经部署好了nacos&#xff0c;我们就可以使用nacos做配置中心和注册中心了。 一、配置中心 有了nacos&#xff0c;我们在微服务项目的配置文件里只需要做一些简单的配置就行了&#xff1a;服务名、服务端口、nacos的地址。其余的配置都可以用…

DGCNN代码详解(一)

以下是 knn 和 get_edge_feature 函数的逐行解释: knn 函数 def knn(x, k):inner = -2 * torch.matmul(x.transpose(2, 1), x) # (B, N, N)计算点云之间的内积,用于计算成对距离。 x.transpose(2, 1) 转置张量以便矩阵乘法。 结果是一个大小为 (B, N, N) 的张量。 xx = to…

TCP/UDP通信协议

TCP通讯时序 下图是一次TCP通讯的时序图。TCP连接建立断开。包含大家熟知的三次握手和四次挥手。 在这个例子中&#xff0c;首先客户端主动发起连接&#xff08;connet&#xff09;、发送请求&#xff0c;然后服务器端响应请求&#xff0c;然后客户端主动关闭连接。两条竖线表…

特斯联|日常|Java|后端开发

2024.9.27 一面 整个沟通下来比较轻松&#xff0c;技术问的比较少比较浅。 自我介绍。实习做的什么项目&#xff1f;介绍一下&#xff1f;用过Vue&#xff1f;前端也会吗&#xff1f;大概有多个页面&#xff1f;介绍一下简历项目&#xff1f;你在项目中用到了Redis和MQ&#…

MySQL中什么情况下类型转换会导致索引失效

文章目录 1. 问题引入2. 准备工作3. 案例分析3.1 正常情况3.2 发生了隐式类型转换的情况 4. MySQL隐式类型转换的规则4.1 案例引入4.2 MySQL 中隐式类型转换的规则4.3 验证 MySQL 隐式类型转换的规则 5. 总结 如果对 MySQL 索引不了解&#xff0c;可以看一下我的另一篇博文&…