gcc编译链接原理及使用

gcc 的使用方法:

gcc    【选项】    文件名

gcc常用选项:

  -v:查看gcc 编译器的版本,显示gcc执行时的详细过程

  -o    < file >             Place  the output  into   < file >

          指定输出文件名为file,这个名称不能跟源文件名同名  

  -E        Preprocess only; do not compile, assemble or link

           只预处理,不会编译、汇编、链接

  -S        Compile only; do not assemble or link

          只编译,不会汇编、链接

  -c        Compile and assemble, but do not link

           

       

//=======================================

gcc  -v: 查看 gcc 编译器的版本

方式1:

gcc  hello.c      输出一个 a.out,然后 ./a.out 来执行该应用程序

gcc   -o   hello   hello.c     输出hello ,然后 ./hello 来执行该应用程序。

方式2:

gcc  -E   -o  hello.i    hello.c

gcc  -S   -o  hello.s    hello.i

gcc  -c   -o  hello.o   hello.s

gcc   -o   hello    hello.o

.o: object   file (OBJ文件)

小结:

1)输入文件的后缀名和选项共同决定gcc到底执行哪些操作。

2)在编译过程中,除非使用了-E、-S、-C选项(或者编译出错阻止了完整的编译过程)

  否则最后的步骤都是链接。

方式3:

gcc  -c   -o  hello.o   hello.s

gcc   -o   hello    hello.o

gcc会对.c文件默认进行预处理操作, -c再来指明了编译、汇编,从而得到.o文件

再通过gcc  -o  hello   hello.o   将.o文件进行链接,得到可执行应用程序。

链接就是将汇编生成的OBJ文件、系统库的OBJ文件、库文件链接起来,

最终生成可以在特定平台运行的可执行程序。

crt1.o、crti.o、crtbegin.o、crtend.o、crtn.o是gcc加入的系统标准启动文件,

对于一般应用程序,这些启动是必需的。

-lc:链接libc库文件,其中libc库文件中就实现了printf等函数。

gcc  -v  -nostdlib  -o   hello  hello.o 会提示因为没有链接系统标准启动文件和标准库文件,而链接失败。

这个-nostdlib选项常用于裸机 /bootloader 、linux 内核等程序,因为它们不需要启动文件、标准库文件。

一般应用程序才需要系统标准启动文件和标准库文件。、

裸机 /BootLoader、linux内核灯程序不需要启动文件、标准库文件。

动态链接使用动态链接库进行链接,生成的程序在执行的时候需要加载所需的动态库才能巡行。

动态链接生成的程序体积较小,但是必须依赖所需的动态库,否则无法执行。

静态链接使用静态库进行链接,生成的程序包含程序运行所需要的全部库,可以直接运行,

不过静态链接生成的程序比较大。

gcc  -c  -o  hello.o  hello.c

gcc   -o  hello_shared  hello.o

gcc   -static  -o  hello_static  hello.o

2、举例说明编译链接过程

一个C/C++文件要经过预处理(preprocessing)、编译(compilation)、汇编(assembly)、和连接(linking)才能变成可执行文件。

以下列程序为例,追层来分析编译过程。 
hello.c:

#include <stdio.h>

#define   MAX  20
#define   MIN  10 

#define  _DEBUG
#define   SetBit(x)  (1<<x) 

int main(int argc, char* argv[])
{
    printf("Hello World \n");
    printf("MAX = %d,MIN = %d,MAX + MIN = %d\n",MAX,MIN,MAX + MIN); 

#ifdef _DEBUG
    printf("SetBit(5) = %d,SetBit(6) = %d\n",SetBit(5),SetBit(6));
    printf("SetBit( SetBit(2) ) = %d\n",SetBit( SetBit(2) ));
#endif
    return 0;
}

① 预处理:

gcc -E -o hello.i hello.c

  • 预处理就是将要包含(include)的文件插入原文件中、将宏定义展开、根据条件编译命令选择要使用的代码,最后将这些代码输出到一个“.i”文件中等待进一步处理。


② 编译:

gcc -S -o hello.s hello.i

  • 编译就是把C/C++代码(比如上面的”.i”文件)“翻译”成汇编代码。

③ 汇编:

gcc -c -o hello.o hello.s

.o:object file(OBJ文件) 这里表现为二进制目标文件: 

  • 汇编就是将第二步输出的汇编代码翻译成符合一定格式的机器代码,在Linux系统上一般表现位ELF目标文件(OBJ文件)。

④ 链接:

gcc -o hello hello.o

  • 链接就是将汇编生成的OBJ文件、系统库的OBJ文件、库文件链接起来,最终生成可以在特定平台运行的可执行程序。


总结:在编译过程中。除非使用了”-c”,“-S”,或”-E”选项(或者编译错误阻止了完整的过程),否则统一完整链接步骤。

譬如:gcc hello.c 和gcc -o hello hello.c都已经完成链接操作。 

又如:gcc -c -o hello.o hello.c 

链接原理:

gcc -c -o hello.o hello.c 不作最后一步链接,得到hello.o二进制OBJ文件

gcc -v -o hello hello.o 我们来看一样链接过程是怎样的:


  • crt1.o、crti.o、crtbegin.o、crtend.o、crtn.o是gcc加入的系统标准启动文件,对于一般应用程序,这些启动是必需的。
  • -lc:链接libc库文件,其中libc库文件中就实现了printf等函数。

① 动态链接:动态链接使用动态链接库进行链接,生成的程序在执行的时候需要加载所需的动态库才能运行。  动态链接生成的程序体积较小,但是必须依赖所需的动态库,否则无法执行。

默认使用动态链接:gcc -o hello_shared hello.o

② 静态链接:静态链接使用静态库进行链接,生成的程序包含程序运行所需要的全部库,可以直接运行,不过静态链接生成的程序体积较大。

gcc -static -o hello_static hello.o

③ -nostartfiles

不链接系统标准启动文件,而标准库文件仍然正常使用: 
gcc -v -nostartfiles -o hello hello.o


④ -nostdlib(最常用)

不链接系统标准启动文件和标准库文件: 
gcc -v -nostdlib -o hello hello.o


- 会提示因为没有链接系统标准启动文件和标准库文件,而链接失败。 
- 这个-nostdlib选项常用于裸机/bootloader、linux内核等程序,因为它们不需要启动文件、标准库文件。

原文地址:https://www.cnblogs.com/piterzhang/p/8971617.html

时间: 2024-07-30 01:09:50

gcc编译链接原理及使用的相关文章

C++常见gcc编译链接错误解决方法

除非明确说明,本文内容仅针对x86/x86_64的Linux开发环境,有朋友说baidu不到,开个贴记录一下(加粗字体是关键词): 用“-Wl,-Bstatic”指定链接静态库,使用“-Wl,-Bdynamic”指定链接共享库,使用示例:-Wl,-Bstatic -lmysqlclient_r -lssl -lcrypto -Wl,-Bdynamic -lrt -Wl,-Bdynamic -pthread -Wl,-Bstatic -lgtest("-Wl"表示是传递给链接器ld的参数

GCC编译链接过程

编译链接过程 代码 #cat main.c #include <stdio.h> int add(int x, int y); int sub(int x, int y); int mul(int x, int y); int div(int x, int y); int main(void) { printf("add:%d\n", add(1,2)); printf("sub:%d\n", sub(10,100)); printf("mul

GCC编译命令常用选项

GCC是GUN Compiler Collection的简称,除编译程序外,还包含其他相关工具.GCC可将高级语言编写的源代码构建成计算机直接执行的二进制代码.GCC是Linux平台下最常用的编译程序,也是Linux平台编译器的事实标准.GCC支持四十余种不同目标体系结构(如X86系列.ARM及PowerPC等),可运行在不同的操作系统上(如Linux.Solaris及Windows等).GCC既支持基于宿主的开发(即在A平台上编译该平台使用的程序),也支持交叉编译(即在A平台上编译供B平台使用

GCC编译器编译链接

在gcc编译器环境下,常见的文件扩展名的含义如下: .c:C源程序,经过预编译后的源程序也为.c文件,它可以通过-E参数输出. .h:头文件 .s:经过编译得到的汇编程序代码,它可以通过-S参数输出. .o:目标文件 .a:函数库 Gcc编译器常见语法: -c:只进行编译,不进行链接,输出的是与源文件同名的.o文件. -o:指定生成的文件的名称.链接生成可执行文件,这个参数后可以带可执行文件的名字,如果没有指定可执行文件的名字,则会默认为a.out. -S:输出汇编代码文件,输出一个与源文件同名

【转】关于编译链接——gcc/g++

添加运行时共享库目录 运行使用共享库的程序需要加载共享库(不同于G++ 编译时指定的链接库),添加共享库的步骤: 修改文件 /etc/ld.so.conf 添加共享库目录 运行 ldconfig 同步更新一下 如: $ gedit /etc/ld.so.conf #添加 /root/dreamlove/lib $ ldconfig 添加include,lib的搜寻路径 对所有用户有效修改/etc/profile 对个人有效则修改~/.bashrc #在PATH中找到可执行文件程序的路径. exp

gcc 编译和链接

1.现在对两个文件生成可执行文件 //thanks.c #include <stdio.h> int main(void) { printf("Hello World\n"); thanks_2(); } //thanks_2.c #include <stdio.h> int thanks_2(void) { printf("Hello World2\n"); return (1); } 2.步骤 一.cd 到文件所在目录 二.gcc -c

[转载]GCC 编译使用动态链接库和静态链接库--及先后顺序----及环境变量设置总结

来自http://blog.csdn.net/benpaobagzb/article/details/51364005 GCC 编译使用动态链接库和静态链接库 1 库的分类 根据链接时期的不同,库又有静态库和动态库之分. 静态库是在链接阶段被链接的(好像是废话,但事实就是这样),所以生成的可执行文件就不受库的影响了,即使库被删除了,程序依然可以成功运行. 有别于静态库,动态库的链接是在程序执行的时候被链接的.所以,即使程序编译完,库仍须保留在系统上,以供程序运行时调用.(TODO:链接动态库时链

gcc编译的四个阶段:预处理,编译,汇编,链接

1:gcc编译的四个阶段:预处理,编译,汇编,链接 #vi file.c #gcc -E file.c -o file.i//-E查看且预处理后停止编译,-o生成目标文件,-i表示已预处理 #gcc -S file.i -o file.s//-S编译到汇编而不进行汇编和链接 #gcc -c file.s -o file.o//-c编译到目标代码 #gcc file.o -o file//-o 文件输出到文件 # gcc -static file.c -o file//-static禁止使用动态库

GCC 编译过程和链接库

GCC 加工程序的过程 在Linux下进行C语言编程,必然要采用GNU GCC来编译C源代码生成可执行程序. 一.GCC使用 Gcc指令的一般格式为:Gcc [选项] 要编译的文件 [选项] [目标文件] 其中,目标文件可缺省,Gcc默认生成可执行的文件名为:a.out 我们来看一下经典入门程序"Hello World!" # vi  main.c #include <stdio.h> void main () { printf("hello world!\n&q