Linux基础之gcc

前言

在Linux开发环境中,比较常用的C\C++的编译器当属gcc(GNU Compiler Collection)了,当然它能够编译的不止C\C++,它不仅能够编译多种语言,而且还是一个交叉编译的平台编译器,所以很有必要好好学习一下的。

安装

在ubuntu 12.4下,gcc是默认安装的,如果系统未安装gcc,安装也是很简单的:

$sudo apt-get install gcc

还需要安装常用的头文件和库文件:

$sudo apt-get install build-essential

完成,可以通过使用一下gcc -v查看gcc版本。

$gcc -v
Target: x86_64-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Ubuntu/Linaro 4.6.3-1ubuntu5' --with-bugurl=file:///usr/share/doc/gcc-4.6/README.Bugs --enable-languages=c,c++,fortran,objc,obj-c++ --prefix=/usr --program-suffix=-4.6 --enable-shared --enable-linker-build-id --with-system-zlib --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --with-gxx-include-dir=/usr/include/c++/4.6 --libdir=/usr/lib --enable-nls --with-sysroot=/ --enable-clocale=gnu --enable-libstdcxx-debug --enable-libstdcxx-time=yes --enable-gnu-unique-object --enable-plugin --enable-objc-gc --disable-werror --with-arch-32=i686 --with-tune=generic --enable-checking=release --build=x86_64-linux-gnu --host=x86_64-linux-gnu --target=x86_64-linux-gnu
Thread model: posix
gcc version 4.6.3 (Ubuntu/Linaro 4.6.3-1ubuntu5) 

编译简介

gcc的编译程序的分为四个阶段:

  • 预处理(pre-processing)
  • 编译(compiling)
  • 汇编(assembing)
  • 连接(linking)

因为在大多数情况下,我们并不需要中间过程,所以四个过程直接整合成一个,直接得到的是可执行程序。

以一个简单的c语言程序为例说明:

  1. 创建c程序,来个最简单的hello world
#include <stdio.h>
int main(void){
   printf("Hello World!");
   return 0;
}
  1. 编译
$gcc hello.c -o hello

编译生成hello这个可执行程序。

  1. 执行
$./hello

得到输出结果Hello world!

编译分解

  1. 预处理
    预处理阶段负责处理预处理命令,例如其中的#include、#define等。
    使用-E可以使编译停在预处理结果:
$gcc -E hello.c -o hello.i

查看生成的hello.i会发现stdio.h被实际的文件内容所替代,宏定义等预处理也被实际内容所替代。

  1. 编译
    编译阶段是将预处理完成的程序编译成汇编语言。
$gcc -S hello.i -o hello.s

查看生成的hello.s,发现已经变成汇编语言:

        .file   "hello.c"
        .section        .rodata
       .LC0:
        .string "Hello World!"
        .text
        .globl  main
        .type   main, @function
       main:
       .LFB0:
        .cfi_startproc
        pushq   %rbp
        .cfi_def_cfa_offset 16
        .cfi_offset 6, -16
        movq    %rsp, %rbp
        .cfi_def_cfa_register 6
        movl    $.LC0, %eax
        movq    %rax, %rdi
        movl    $0, %eax
        call    printf
        movl    $0, %eax
        popq    %rbp
        .cfi_def_cfa 7, 8
        ret
        .cfi_endproc
       .LFE0:
        .size   main, .-main
        .ident  "GCC: (Ubuntu/Linaro 4.6.3-1ubuntu5) 4.6.3"
        .section        .note.GNU-stack,"",@progbits
  1. 汇编
    在汇编阶段,gcc把汇编的代码编译成CPU可以执行的机器码,也就是目标代码模块。
$gcc -c hello.s -o hello.o
  1. 连接
    连接截断把生成的目标模块以及各种库文件连接在一起,形成一个二进制的可执行文件。
$gcc hello.o -o hello

得到的可执行文件就可以运行了。

多目标编译

通常情况下,为了实现模块化编程的目的,一个程序通常会被分成若干个源文件,这时当然不需要每一个文件都编译一遍再连接到一起,gcc会自动帮你完成这些事情。

$gcc file1.c file2.c file3.c -o hello 

当然它其实相当于每个分别编译然后连接了:

$gcc file1.c -o file1.o
$gcc file2.c -o file2.o
$gcc file3.c -o file3.o
$gcc file1.o file2.o file3.o  -o hello

当然更复杂的情况可能需要使用make这个工具去完成了。

    原文作者:小时不识月z
    原文地址: https://www.jianshu.com/p/e87b8a7af2b3
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞