gcc编译链接原理及使用

摘要:
方法3:ogcc-c-ohello.hello.sgc-ohellhello.sgcc将预处理。默认情况下,-c文件将指示编译和汇编,因此。o文件,然后获取。o文件可以通过gcc-ohellohellhello.o链接,以获得可执行的应用程序。gcc-v-nostdlb ohellohello你好。O系统标准启动文件和标准库文件未链接,链接失败。默认情况下使用动态链接:gcc-ohello_sharedhello。O② 静态链接:静态链接使用静态库进行链接,生成的程序包含程序操作所需的所有库,可以直接运行,但静态链接生成的程序体积较大。
1、gcc 和 arm-linux-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到底执行哪些操作。

gcc编译链接原理及使用第1张

gcc编译链接原理及使用第2张

gcc编译链接原理及使用第3张

 

gcc编译链接原理及使用第4张

 

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 
");
    printf("MAX = %d,MIN = %d,MAX + MIN = %d
",MAX,MIN,MAX + MIN); 

#ifdef _DEBUG
    printf("SetBit(5) = %d,SetBit(6) = %d
",SetBit(5),SetBit(6));
    printf("SetBit( SetBit(2) ) = %d
",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内核等程序,因为它们不需要启动文件、标准库文件。

免责声明:文章转载自《gcc编译链接原理及使用》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇JFreeChart画折线图C++与C#的时间转换下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

Linux使用unzip打开zip文件失败提示无法打开压缩包:Not zip archive

0x01问题概述: 常规解压使用UNzip,但有时会报错。目前UNzip最大可解压2G的文件 0x02解决方法 环境检测 首先确保下列程序已安装 如果不是因为程序缺失那执行第二步 检查gcc-c++ yum list | grep gcc 检查gcc-c++ 安装unzip yum install unzip 一条命令开始解压:sudo apt de...

ubuntu 14.04 源码编译mysql-5.7.17

环境为 Ubuntu 12.04 64 位的桌面版 编译的mysql 版本为 5.7.18 首先需要安装一下依赖包 sudo apt-get install libncurses5-dev cmake chkconfig 创建用户组和用户 mysql groupadd mysql useradd -r -g mysql -s /bin/bash mysq...

Linux makefile 教程 非常详细,且易懂

Linux makefile 教程 非常详细,且易懂 陈皓 (CSDN) 概述—— 什么是makefile?或许很多Winodws的程序员都不知道这个东西,因为那些Windows的IDE都为你做了这个工作,但我觉得要作一个好的和professional的程序员,makefile还是要懂 。这就好像现在有这么多的HTML的编辑器,但如果你想成为一个专业人士,...

关于编译Qt以及驱动的一点总结吧 Rollen Holt 博客园

关于编译Qt以及驱动的一点总结吧 - Rollen Holt - 博客园 关于编译Qt以及驱动的一点总结吧 通过这几次的折腾,对编译Qt算是有一点的心得吧: 首先我们编译QT的时候需要面对的是下载源代码,一般情况下我们建议下载qt-everywhere-opensource-src-4.8.2.zip这种形式的源代码。这样编译起来的问题也就很少了。当然...

Response.flush的用法

很多时候我们写的程序,会花上一分钟甚至几分钟时间。为了使软件使用者能够耐心的等待程序的执行,我们经常会希望有一个进度条来表示程序执行的状态。或者最起码要显示一个类似: “数据载入中”,“正在保存数据” 等的说明性文字。此时我们就会用到Response.flush()(刷新缓冲区数据)。他会将缓冲区中编译完成的数据先发送到客户端。 (引用: 但是有很多时候,...

嵌入式web服务

:boa、thttpd、mini_httpd、shttpd、lighttpd、goaheand、appweb和apache等。 Boa 1.介绍 Boa诞生于1991年,作者Paul Philips。是开源的,应用很广泛,特别适合于嵌入式设备,网上流行程度很广。它的官方网站说boa是最受人喜爱的嵌入式web服务器。功能较为强大,支持认证,cgi等。B...