GCC-GCC编译流程

序言

对于大多数程序员而言,大家都知道gcc是什么,但是如果不接触到linux平台下的开发,鲜有人真正了解gcc的编译流程,因为windows+IDE的开发模式简直是一条龙全套服务,开发者只需要关系代码逻辑与功能实现即可,但是,在享受便利的同时,必然也牺牲了一些灵活性。

gcc是什么

国际惯例,先得介绍gcc是什么,gcc的原名为GNU C Compiler,专门针对C语言的编译器,而在后来计算机的发展中,GCC逐渐兼容了C++,java等语言,发展为扩展版的GCC,全称为 GNU compiler collection,事实上它是指一套编译器,而不再是单纯的C编译器,像g++,其实也是属于GCC工具中的一种。

gcc的基本使用

gcc的编译过程分多步完成,基本可分为4步:预编译,编译,汇编,链接,gcc的一般语法为

gcc [-option] <file> [-option] [-dst file]

举个例子,如果需要编译一个hello_world.c文件

gcc hello_world.c -o hello_world

在上述例子中,选择默认的编译选项,-o 为指定可执行文件的名称,上述生成hello_world可执行文件,如果不指定可执行文件名称,默认生成a.out可执行文件。

进入hello_world所在目录,即可使用./hello_world指令运行程序

常用gcc编译选项

-E

gcc的-E选项只对目标文件进行预编译处理

-S

gcc的-S选项只对目标文件进行预编译处理和编译处理,这里的编译处理仅仅是将预编译处理后的文件编译成汇编文件

-C

gcc的-C选项对目标文件进行预编译,编译,汇编处理,生成二进制目标文件

在默认情况下,gcc编译时执行预编译,编译,汇编和链接过程,直接生成可执行代码

-D

gcc的-D选项相当于在源文件中全局添加一个宏定义,一般在平台兼容或者程序模式切换中比较常见,例如:

#ifdef ARM
DO SOMETHING
#elif X86
DO SOMETHING
....
#endif

这时候在编译时或者在Makefile中加入 -DARM 或者 -DX86 来选择平台,而不用改源代码。

又或者,如果你在开发过程中,需要调试某些功能,经常会写一些调试代码,而有时候又想屏蔽部分调试代码,就可以这样实现:

#ifdef DEBUG_PART1
DO SOMETHING
#ifdef DEBUG_PART2
DO SOMETHING
...
#endif

-O(n)

gcc的-O选项规定了程序的优化等级,分为三个等级,分别是-O,-O2,-O3,所对应的优化等级依次增高.

gcc提供代码优化功能,可以基本做到同时优化程序空间和运行效率,但是并不是优化等级越高越好,因为优化等级越高就越可能出现一些潜在的风险,因为gcc的优化并不总是正确的,同时可能改变程序逻辑结构,加大调试难度,一个比较普遍的建议是在空间和效率可承受范围内选最低的优化等级。

-std=XXX

gcc的-std=XXX,这个XXX指定了gcc编译器的版本,如-std=c99或者-std=c++11,因为在默认编译条件下,可能不支持某些语言新版本的特性,当程序需要用到高版本语言特性时,则需要指定编译时对应的语言标准

gcc编译流程

预编译

预编译指将文件中包含的头文件展开,将所有的宏进行替换,同时将所有的条件编译解析出来添加到文件中。

所对应的编译指令为:

gcc -E file >> dst.i

需要注意的是,预编译并不生成相应的中间文件,直接输出到终端,因此需要将结果重定向到文件中以便查看。一个简单的hello_world.c文件经过预编译的过程将产生800多行的代码,我将在另一篇博客中详细讨论预编译中的宏、和条件编译。

编译

这个编译过程是我们通常讲的程序编译的一部分,是将经过预编译处理的代码编译成汇编代码。

所对应的编译指令为:

gcc -S file -o dst.s

汇编

汇编过程是将汇编语言的文件编译成目标文件,即二进制文件,我们通常所使用的静态库和动态库

编译指令为:

gcc -c file -o dst.o

链接

经过汇编生成的目标文件,如果是多个文件的编译,将会生成多个目标二进制文件,ld连接器将所有的二进制文件链接起来,根据平台的不同添加上不同的头部信息,分配内存空间,需要注意的是,在所有需要链接的目标文件中,有且仅有一个main()函数,main()函数作为程序入口函数只能有一个,否则会报错。

在链接过程中,还将涉及到链接静态库和动态库的问题。

很多朋友对库的概念应该不算陌生,经常会用到各种各样的库,但是仅仅停留在理解的层次而已,甚至都从来没有手动管理过库,其实可以说基本上每个程序员写的每个程序都要用到库函数,在C/C++语言中,最明显的例子就是包含头文件

#include <stdio.h>

其实,这个包含头文件的动作就是声明了glibc中的库函数,表明我的程序中需要调用到glibc中的库,而在程序进行链接的时候将会将glibc库链接进程序,但是这里有一个动态库和静态库的区分:

静态库:直接在链接阶段就将库文件链接到可执行文件中,最明显的缺点是可执行文件的尺寸变大。

动态库:在程序运行时才被载入,相对于静态库而言,就有几个明显的优点:

  1. 优化了静态库的浪费空间问题
  2. 由于只需要在内存中存在一个库文件,所有程序可以共享,所以在动态库后续的维护升级中,不必去一个个修改应用程序,而只需要修改库中的内容,降低耦合性。

gcc链接指令

-L 指定链接库的目录

-l 指定需要链接的库名称

接下来的博客会详细讲解静态库和动态库的生成以及使用。

(完)

好了,关于GCC编译流程的问题就到此为止了,如果朋友们对于这个有什么疑问或者发现有文章中有什么错误,欢迎留言

原创博客,转载请注明出处!

祝各位早日实现项目丛中过,bug不沾身.

(完)

GCC 编译流程简介的更多相关文章

  1. GCC编译流程浅析

    GCC-GCC编译流程浅析 序言 对于大多数程序员而言,大家都知道gcc是什么,但是如果不接触到linux平台下的开发,鲜有人真正了解gcc的编译流程,因为windows+IDE的开发模式简直是一条龙 ...

  2. GCC编译流程及常用编辑命令

    GCC 编译器在编译一个C语言程序时需要经过以下 4 步: 将C语言源程序预处理,生成.i文件. 预处理后的.i文件编译成为汇编语言,生成.s文件. 将汇编语言文件经过汇编,生成目标文件.o文件. 将 ...

  3. gcc编译流程

    gcc的编译流程分为四个步骤,分别为: 预处理(Pre-Processing) 编译(Compiling) 汇编(Assembling) 链接(Linking) 以hello.c为例子,在这四个步骤中 ...

  4. gcc 编译流程分析

    //test.c #include<stdio.h> int main() { ,y=; printf("x=%d y=%d\n",x,y); ; } 1:预处理阶段, ...

  5. 面试题----gcc的编译流程

    gcc编译流程 一.    编译与处理指令: gcc -E hello.c -o a.c 如果不使用-o指定输出的文件,会默认输出到终端.所以建议使用同时使用-o选项. 还要注意:编译时会保留#pra ...

  6. gcc编译工具生成动态库和静态库之一----介绍

     1.库的分类 根据链接时期的不同,库又有静态库和动态库之分. 静态库是在链接阶段被链接的(好像是废话,但事实就是这样),所以生成的可执行文件就不受库的影响了,即使库被删除了,程序依然可以成功运行. ...

  7. gcc编译工具生成动态库和静态库

    一. 库的分类 1.1. 静态库(.a) 1.1.1. 静态库的代码在编译过程中已经被载入可执行程序,因此体积比较大.所以生成的可执行文件就不受库的影响了,即使库被删除了,程序依然可以成功运行. 1. ...

  8. Gcc的编译流程分为了四个步骤:

    http://blog.csdn.net/xiaohouye/article/details/52084770(转) Gcc的编译流程分为了四个步骤: 1.预处理,生成预编译文件(.文件): Gcc ...

  9. gcc 的编译流程 和gdb的调试方法

    GCC的编译流程分为四个步骤: 预处理(Pre-Processing) 编译(Compiling) 汇编(Assembling) 链接(Linking) 可以看的出来文件大小 gdb 调试 gdb - ...

随机推荐

  1. 【LeetCode算法-28/35】Implement strStr()/Search Insert Position

    LeetCode第28题 Return the index of the first occurrence of needle in haystack, or -1 if needle is not ...

  2. [LeetCode] 188. Best Time to Buy and Sell Stock IV 买卖股票的最佳时间 IV

    Say you have an array for which the ith element is the price of a given stock on day i. Design an al ...

  3. [LeetCode] 237. Delete Node in a Linked List 删除链表的节点

    Write a function to delete a node (except the tail) in a singly linked list, given only access to th ...

  4. k8s pv,pvc无法删除问题

    一般删除步骤为:先删pod再删pvc最后删pv 但是遇到pv始终处于“Terminating”状态,而且delete不掉.如下图: 解决方法: 直接删除k8s中的记录: 1 kubectl patch ...

  5. 【神经网络与深度学习】【CUDA开发】服务器(多GPU)caffe安装和编译

    一. 前提 多GPU交互在神经网络是常见的,所以在安装caffe之前需要安装NCCL,来保证多GPU之间的相互交流.  多GPU,这里指的是2个及2个以上英伟达显卡,而不是笔记本中的集显和独显. 二. ...

  6. Linux安装卸载JDK完整步骤

    1.检查一下系统中的jdk版本 [root@localhost software]# java -version 显示: openjdk version "1.8.0_102" O ...

  7. idea关闭sonarLint自动扫描

    手动运行SonarLint 停止SonarLint自动检测代码之后,可以使用Ctrl+Shift+S手动运行SonarLint检查代码

  8. Centos7挂载新硬盘

    1.查看系统是否检测到新的硬盘设备 ls /dev/ |grep sd linux 中所有外设都会在这个目录下,对应一个文件,其中第一块硬盘是sda,第二块硬盘是sdb,第三块硬盘是sdc.其中sda ...

  9. TCP/IP学习笔记12-- IP协议基础

    IP(internet protocol, 网际协议) IP相当于OSI参考模型中的第三层 -- 网络层. --网络层的下一层--数据链路层的主要作用是在 同一种数据链路 的节点之间进行包传递,而一旦 ...

  10. ubuntu12下安装unixODBC(mysql)

    转自:https://blog.51cto.com/dreamylights/1321678 1. 需要的包 unixODBC源码包unixODBC-2.2.14.tar.gz mysql 驱动 my ...