gcc 编译一个C文件的过程分析,动态链接生成

本文详细介绍了GCC编译器的工作原理及其编译过程中的四个主要阶段:预处理、编译、汇编和链接。通过实际例子展示了每个阶段的命令选项及输出结果。

#gcc hello.c

该命令将hello.c直接生成最终二进制可执行程序a.out

这条命令隐含执行了(1)预处理、(2)汇编、(3)编译并(4)链接形成最终的二进制可执行程序。这里未指定输出文件,默认输出为a.out。

 

GCC编译C源码有四个步骤:

预处理-----> 编译 ----> 汇编 ----> 链接

现在我们就用GCC的命令选项来逐个剖析GCC过程。

1)预处理(Pre-processing)

  在该阶段,编译器将C源代码中的包含的头文件如stdio.h编译进来,用户可以使用gcc的选项”-E”进行查看。

用法:#gcc -E hello.c -o hello.i

作用:将hello.c预处理输出hello.i文件。

选项 -S

用法:[root]# gcc –S hello.i –o hello.s

作用:将预处理输出文件hello.i汇编成hello.s文件。

[root@richard hello-gcc]# ls

hello.c  hello.i  hello.s

3)汇编阶段(Assembling)

  汇编阶段是把编译阶段生成的”.s”文件转成二进制目标代码.

选项 -c

用法:[root]# gcc –c hello.s –o hello.o

作用:将汇编输出文件test.s编译输出test.o文件。

[root]# gcc -c hello.s -o hello.o

[root]# ls

hello.c  hello.i  hello.o  hello.s

4)链接阶段(Link)

无选项链接

用法:[root]# gcc hello.o –o hello.exe

作用:将编译输出文件hello.o链接成最终可执行文件hello.exe。

[root]# ls

hello.c  hello.exe  hello.i  hello.o  hello.s

[root@localhost Gcc]# ./hello

Hello World!

在这里涉及到一个重要的概念:函数库。

[root]# ldd hello.exe

libc.so.6 => /lib/tls/libc.so.6 (0x42000000)

/lib/ld-linux.so.2 => /lib/ld-linux.so.2 (0x40000000)

函数库一般分为静态库和动态库两种。静态库是指编译链接时,把库文件的代码全部加入到可执行文件中,因此生成的文件比较大,但在运行时也就不再需要库文件了。其后缀名一般为”.a”。动态库与之相反,在编译链接时并没有把库文件的代码加入到可执行文件中,而是在程序执行时由运行时链接文件加载库,这样可以节省系统的开销。动态库一般后缀名为”.so”,如前面所述的libc.so.6就是动态库。gcc在编译时默认使用动态库。

 

test.h

  1. #ifndef  _TEST_H  
  2. #define  _TEST_H  
  3.   
  4. int add(int a, int b);  
  5. #endif  

 

test.c

  1. #include "test.h"  
  2.   
  3. int add(int a, int b)  
  4. {  
  5.     return a + b;  

我们想要生成动态库,要做的工作其实非常简单,输入gcc -shared -fPIC -o libtest.so test.c即可

hello.c

  1. #include <stdio.h>  
  2. #include "test.h"  
  3.   
  4. int main()  
  5. {  
  6.     printf("%d\n", add(2, 3));  
  7.     return 1;  
  8. }

 

输入gcc hello.c -o  hello ./libtest.so。然后输入./hello

就行了。

 

 

 

 

 

 

 

 

 

<think>好的,我现在需要回答用户关于GCC编译C文件生成的目标产物的问题。首先,我需要回忆GCC编译流程和相关知识,并参考提供的引用内容来确保准确性。 用户的问题是关于GCC编译C文件最终生成文件格式。根据我的理解,GCC编译过程分为预处理、编译、汇编和链接四个阶段,每个阶段会生成不同的中间文件,最终生成可执行文件。但用户可能想知道具体的文件格式,比如目标文件和可执行文件的格式。 首先,参考引用中的内容。引用4提到,预处理生成.i文件编译生成.s汇编文件,汇编生成.o目标文件,最后链接生成可执行二进制文件。引用2和5也提到目标文件.o文件,可执行文件没有后缀名或者可能有自定义的后缀。引用3提到生成可执行文件的步骤。 需要明确的是,在Linux系统下,GCC默认生成的可执行文件通常是ELF(Executable and Linkable Format)格式,但用户可能更关心文件扩展名。目标文件.o,可执行文件一般没有扩展名,如使用gcc -o test test.c生成名为test的可执行文件。 另外,用户可能想知道不同阶段生成的具体文件类型,比如预处理后的.i文件编译后的.s文件,汇编后的.o文件,以及链接后的可执行文件。需要分阶段说明,并指出各阶段的产物格式。 确认引用中的信息是否准确。例如,引用5提到汇编生成中间目标文件,使用-c选项生成.o文件。这与ELF格式的目标文件一致。引用4详细描述了每个步骤的输出文件,因此可以引用这些内容来支持回答。 需要注意的是,不同的操作系统生成的可执行文件格式可能不同,但用户的问题是在Linux环境下,因为引用了linux gcc编译的内容(引用4)。因此,可执行文件格式应为ELF,而目标文件是ELF格式的.o文件。 总结回答结构:首先分阶段说明每个步骤生成文件类型和格式,然后总结最终产物。需要引用相关的引用资料来支持每个阶段的描述,比如预处理后的.i文件(引用4),汇编后的.o文件(引用2和5),链接后的可执行文件(引用3和4)。 可能还需要提到,如果不进行链接,仅编译到目标文件,则生成.o文件;如果完整编译,则生成可执行文件。例如,gcc -c会生成.o,而gcc直接生成可执行文件。 最后,确保回答符合用户的要求,使用中文,正确使用LaTeX格式,生成相关问题,并正确引用参考资料。例如,在提到预处理阶段时引用引用4,汇编阶段引用5等。 现在需要组织语言,按照流程分步骤解释,并引用对应的引用,确保每个关键点都有支持。同时,生成的相关问题需要围绕GCC编译过程、各阶段产物、文件格式等展开。</think>使用GCC编译C文件的最终产物取决于编译的流程阶段。完整的编译过程分为预处理、编译、汇编和链接四个步骤,每个步骤会生成不同的文件: 1. **预处理阶段** 通过`gcc -E`生成扩展名为`.i`的预处理文件。此阶段会展开头文件、宏替换等。例如: ```bash gcc -E main.c -o main.i ``` 生成的`main.i`是纯C代码文件,但已包含所有宏和头文件内容[^4]。 2. **编译阶段** 通过`gcc -S`生成扩展名为`.s`的汇编代码文件。例如: ```bash gcc -S main.i -o main.s ``` 此阶段将C代码转换为平台相关的汇编指令。 3. **汇编阶段** 通过`gcc -c`生成扩展名为`.o`(或`.obj`)的**目标文件**。例如: ```bash gcc -c main.s -o main.o ``` 目标文件二进制格式(通常是ELF格式),包含机器码和符号表,但未完成链接[^2][^5]。 4. **链接阶段** 最后一步通过`gcc`链接所有目标文件和库,生成**可执行文件**。例如: ```bash gcc main.o -o main ``` 在Linux系统下,默认生成的可执行文件为ELF格式且无扩展名(如`main`);Windows下为PE格式,通常以`.exe`结尾[^3]。 ### 总结 - **目标文件**:扩展名为`.o`,ELF格式,包含未链接的机器码[^5]。 - **可执行文件**:无扩展名(Linux)或`.exe`(Windows),ELF或PE格式,可直接运行。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值