在Linux下进行C语言编程,必然要采用GNU GCC来编译C源代码生成可执行程序。
一、GCC快速入门
Gcc指令的一般格式为:Gcc [选项] 要编译的文件 [选项] [目标文件]
其中,目标文件可缺省,Gcc默认生成可执行的文件名为:编译文件.out
我们来看一下经典入门程序"Hello World!"
# vi hello.c
#include <stdlib.h>
#include <stdio.h>
void main(void)
{
printf("hello world!\r\n");
}
用gcc编译成执行程序。
#gcc hello.c
该命令将hello.c直接生成最终二进制可执行程序a.out
这条命令隐含执行了(1)预处理、(2)汇编、(3)编译并(4)链接形成最终的二进制可执行程序。这里未指定输出文件,默认输出为a.out。
如何要指定最终二进制可执行程序名,那么用-o选项来指定名称。比如需要生成执行程序hello.exe
那么
#gcc hello.c -o hello.exe
二、GCC的命令剖析--四步走
从上面我们知道GCC编译源代码生成最终可执行的二进制程序,GCC后台隐含执行了四个阶段步骤。
GCC编译C源码有四个步骤:
预处理-----> 编译 ----> 汇编 ----> 链接
现在我们就用GCC的命令选项来逐个剖析GCC过程。
1)预处理(Pre-processing)
在该阶段,编译器将C源代码中的包含的头文件如stdio.h编译进来,用户可以使用gcc的选项”-E”进行查看。
用法:#gcc -E hello.c -o hello.i
作用:将hello.c预处理输出hello.i文件。
[root]# gcc -E hello.c -o hello.i
[root]# ls
hello.c hello.i
[root]# vi hello.i
# 1 "hello.c"
# 1 "<built-in>"
# 1 "<command line>"
# 1 "hello.c"
# 1 "/usr/include/stdlib.h" 1 3
# 25 "/usr/include/stdlib.h" 3
# 1 "/usr/include/features.h" 1 3
# 291 "/usr/include/features.h" 3
# 1 "/usr/include/sys/cdefs.h" 1 3
# 292 "/usr/include/features.h" 2 3
# 314 "/usr/include/features.h" 3
# 1 "/usr/include/gnu/stubs.h" 1 3
# 315 "/usr/include/features.h" 2 3
# 26 "/usr/include/stdlib.h" 2 3
# 3 "hello.c" 2
void main(void)
{
printf("hello world!\r\n");
}
2)编译阶段(Compiling)
第二步进行的是编译阶段,在这个阶段中,Gcc首先要检查代码的规范性、是否有语法错误等,以确定代码的实际要做的工作,在检查无误后,Gcc把代码翻译成汇编语言。用户可以使用”-S”选项来进行查看,该选项只进行编译而不进行汇编,生成汇编代码。
选项 -S
用法:[root]# gcc –S hello.i –o hello.s
作用:将预处理输出文件hello.i汇编成hello.s文件。
[root@richard hello-gcc]# ls
hello.c hello.i hello.s
如下为hello.s汇编代码
[root@richard hello-gcc]# vi hello.s
.file "hello.c"
.section .rodata
.LC0:
.string "hello world!\r\n"
.text
.globl main
.type main,@function
main:
pushl %ebp
movl %esp, %ebp
subl $8, %esp
andl $-16, %esp
movl $0, %eax
subl %eax, %esp
subl $12, %esp
pushl $.LC0
call printf
addl $16, %esp
movl $0, %eax
leave
ret
.Lfe1:
.size main,.Lfe1-main
.ident "GCC: (GNU) 3.2.2 20030222 (Red Hat Linux 3.2.2-5)"
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!
在这里涉及到一个重要的概念:函数库。
读者可以重新查看这个小程序,在这个程序中并没有定义”printf”的函数实现,且在预编译中包含进的”stdio.h”中也只有该函数的声明,而没有定义函数的实现,那么,是在哪里实现”printf”函数的呢?最后的答案是:系统把这些函数实现都被做到名为libc.so.6的库文件中去了,在没有特别指定时,gcc会到系统默认的搜索路径”/usr/lib”下进行查找,也就是链接到libc.so.6库函数中去,这样就能实现函数”printf” 了,而这也就是链接的作用。
你可以用ldd命令查看动态库加载情况:
[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在编译时默认使用动态库。
分享到:
相关推荐
以下是arm-linux-gcc编译选项的详细知识点说明: 1. 编译过程的四个阶段: - 预处理阶段:GCC会对源文件进行预处理,展开宏定义、处理条件编译指令、包含头文件等。 - 编译阶段:经过预处理的源文件会被转化为...
标题中的"gcc-arm-none-eabi-9-2019-q4-major-win32.zip"是一个针对Windows操作系统的GCC交叉编译工具链压缩包。这个工具链是专门为ARM架构的微控制器和嵌入式系统设计的,使得开发者可以在运行Windows的个人电脑上...
GCC(GNU Compiler Collection)是GNU项目的核心部分,它不仅包括C、C++、Fortran、Objective-C、Ada等编程语言的编译器,还提供了预处理、汇编、链接等工具。在本案例中,"arm-none-eabi"标识了这是一个针对ARM架构...
4. libexec:这个目录可能包含编译过程所需的辅助程序或者动态链接库,这些文件在编译过程中会被调用,帮助完成诸如预处理、汇编和链接等任务。 5. lib:这个目录可能包含编译器需要的库文件,比如支持不同语言特性...
3. **libexec**:通常存放辅助执行程序或库,这些文件可能在编译过程中被调用,执行一些特定的任务,比如预处理、汇编或优化。 4. **lib**:这个目录下的库文件可能是GCC和G++运行时所需的依赖。在编译和链接阶段,...
Linux中的GCC(GNU Compiler Collection)是开源的、跨平台的编译器套件,用于将C、C++、Fortran、Objective-C等编程语言的源代码编译为可执行文件。GCC 4.8.5是该系列的一个稳定版本,发布于2015年,虽然不是最新版...
预处理处理宏定义、条件编译等,编译将预处理后的文本转换成汇编代码,汇编将汇编代码转化为机器码,最后链接将多个目标文件和库合并成可执行文件。GCC通过源文件的后缀名来判断文件的语言类型,并决定后续处理方式...
交叉编译的基本流程包括:预处理、编译、汇编和链接四个步骤。在这个过程中,编译器会根据指定的目标体系结构生成相应的机器码,而链接器则会结合各种静态和动态库,生成最终的可执行文件。 总之,“gcc-4.6.2-...
它通过一系列命令行参数来控制编译过程的不同方面,包括预处理、编译、汇编和链接等步骤。 ### GCC编译数据库的概念 GCC编译数据库并不是一个传统意义上的数据库,而是一种存储项目编译指令的数据结构或文件格式,...
4. `as`:汇编器,将预处理后的代码转换为机器语言。 5. `ld`:链接器,将编译后的对象文件合并成一个可执行文件或库。 6. 开发头文件和库:提供了Ee500v2平台上的系统调用、硬件接口和其他功能的接口。 在实际开发...
5. 预处理器和汇编器:`cpp`和`as`,分别处理预处理指令和汇编代码。 6. 工具和脚本:用于辅助编译过程,如`make`,用于自动化构建过程。 在Linux环境下安装这个工具包后,开发者可以在本地x86_64系统上构建针对...
在GCC的编译过程中,可以分为四个阶段:预处理(Pre-Processing)、编译(Compiling)、汇编(Assembling)和链接(Linking)。预处理阶段处理#include指令、宏定义和条件编译;编译阶段将预处理后的文本转换成汇编...
编译过程分为预处理、编译、汇编和链接四个步骤。预处理阶段处理宏定义、条件编译指令和包含文件;编译阶段将预处理后的代码转化为汇编语言;汇编阶段将汇编语言转换为机器码;链接阶段则将各个编译生成的目标文件...
2. **编译器** (cc/g++): 编译器如gcc或g++,将预处理过的C/C++源代码转化为汇编代码。 3. **汇编器** (as): 汇编器将汇编代码转化为机器语言的二进制目标文件。 4. **链接器** (ld): 链接器负责合并多个目标文件...
### GCC编译过程详解 #### 一、GCC简介与作用 GCC(GNU Compiler Collection)是GNU发布的一套编程语言编译器,它支持多种编程语言,如C、C++、Objective-C、Fortran等,并且可以针对不同的处理器架构生成优化过的...
GCC 和 G++ 编译器的编译过程可以分为四个步骤:预处理、编译、汇编和链接。预处理阶段将源代码转换为中间代码,编译阶段将中间代码转换为汇编代码,汇编阶段将汇编代码转换为机器代码,链接阶段将机器代码链接成可...
本文将详细讲解Linux下GCC的编译过程,包括预处理、编译、汇编和链接四个阶段,并探讨相关的编译选项和优化原理。 1. 预处理 预处理是GCC处理C源代码的第一步,由cpp程序负责。在这个阶段,GCC执行以下操作: - ...
预处理是GCC编译过程中最初始的阶段。在这个阶段,GCC会处理以`#`字符开头的预处理器指令。预处理器的主要任务包括: - **宏替换**:对宏进行展开,例如`#define`定义的宏会被相应的文本替换。 - **文件包含**:...
G77的编译过程通常涉及预处理、编译、汇编和链接四个步骤,通过g77命令行工具可以完成这些操作。用户需要编写Fortran 77源代码,然后使用g77进行编译,生成可执行文件。 在编程过程中,G77支持Fortran 77标准的特性...