【Linux】gcc编译器的使用(程序的翻译过程)

news/2024/11/8 2:59:13/

目  录

  • 1 程序的翻译
    • 1.1预处理(进行宏替换)
    • 1.2 编译(生成汇编代码)
    • 1.3 汇编(生成机器可识别代码)
    • 1.4 链接(生成可执行文件或者库文件)
    • 1.5 gcc常用选项总结


程序的翻译过程包括:预处理、编译、汇编、链接四个部分,接下来文章中将讨论在Linux下如何使用gcc编译器完成程序的翻译。

1 程序的翻译

gcc使用命令格式:gcc [选项] 要编译的文件 [选项] 生成的目标文件

以下以 test.c 文件作为示例讨论程序翻译过程,其中源代码如下图所示:

test.c


1.1预处理(进行宏替换)

  • 预处理功能主要包括宏定义,文件包含,条件编译,去注释等。
  • 预处理指令是以 # 号开头的代码行。
  • 使用gcc完成预处理过程: gcc -E test.c -o test.i
    其中选项 -E 的作用是让gcc在预处理结束后停止编译过程;选项 -o 是指生成目标文件(注意:-o 选项后面必须紧跟目标文件);.i 文件是经过预处理后的C原始程序,如下图所示,源程序中的注释经过预处理后被删去了,main函数中的N也被替换为了100,还有条件编译也被处理了,此外,还可以看到,除了被处理后的原代码部分,前面还有八百多行代码是头文件等展开后的结果。

test.i


1.2 编译(生成汇编代码)

  • 编译过程包括词法分析、语法分析、语义分析等。在这个过程中,gcc首先要检查代码的规范性、是否有语法错误等,以确定代码实际要做的工作,在检查无误后,gcc把代码翻译成汇编语言程序。
  • 使用gcc完成编译过程:gcc -S test.i -o test.s
    其中 -S 的作用是让gcc在编译结束后停止翻译过程,.s 文件是编译后生成的汇编语言程序(如下图所示)。

test.s


1.3 汇编(生成机器可识别代码)

  • 汇编阶段就是把编译阶段生成的 .s 文件转成目标文件。
  • 使用gcc完成汇编过程:gcc -c test.s -o test.o
    其中 -c 的作用是让gcc在汇编结束后停止翻译过程,.o 文件是汇编后生成的可重定位目标二进制代码(相当于windows下程序翻译后形成的 .obj 文件)。如下图所示,这里只是把我们自己编写的代码翻译成了二进制文件,即使将文件权限修改为可执行后,这个二进制文件仍是不可被执行的,还需经过链接过程形成完整的可执行程序。
    test.o

1.4 链接(生成可执行文件或者库文件)

  • 使用gcc完成链接过程:gcc test.o -o test

链接过程具体做了什么呢?这里需要先了解一个概念:函数库

  • 我们在编写程序时,常常会用到 printf 等库函数,但事实上,在预编译中包含的头文件 stdio.h 中也只是有函数的声明,而没有函数的定义(实现),那 printf 又是在哪里实现的?我们的程序又是如何成功使用函数的呢?
    其实,系统把这些函数实现都写在名为 libc.so.6 的库文件中了,在没有特别指定时,gcc会到系统默认的搜索路径 /usr/lib 下查找,也就是链接到 libc.so.6 库函数中去,如此就能实现对应的库函数了,而这也是链接的作用。
  • 我们之所以能够在Linux下进行C、C++等代码的编写和编译是因为Linux系统默认已经携带了语言级别的头文件和语言对应库,而编译器能够自动帮我们识别解释这些语言,并根据相应的库完成程序的编译链接以形成可执行程序。

函数库一般分为静态库和动态库(共享库)两种,对应的链接也分为静态链接和动态链接

  • 库本质上也是一种文件,Linux下静态库通常以 libXXX.a 的形式命名,动态库通常以 libXXX.so 的形式命名;Windows下静态库通常以 .lib 为后缀,动态库通常以 .dll 为后缀。
  • 静态链接是指在编译链接时,把静态库中我们所需要的代码全部拷贝添加到最终的可执行文件中,因此生成的文件比较大(会比较占用空间),但同时静态链接完成后程序在运行时也就不需要再依赖任何库,自己就可以运行了。
  • 与静态链接不同,动态链接在程序编译链接时并没有把库文件中的代码拷贝添加到可执行文件中,而是将动态库文件中我们所需要的代码的地址拷贝添加到可执行文件中的相关位置,程序运行时则通过地址链接库文件找到相应的代码。如此,所有有需要的程序共享方法,而方法的真正实现永远只在库中,可有效节省系统的开销。但如果相关动态库缺失,对应的程序也将无法执行。

那我们要如何知道我们的可执行程序是静态链接形成的还是动态链接形成的呢?这里首先要了解一个命令:ldd 可执行文件 ,通过该命令可以查看对应可执行文件所依赖的动态库。
以下通过示例来进一步了解程序的链接:

动态链接

  • 需要注意的是:Linux下默认使用的是动态链接和动态库。

那如何实现静态链接呢?

这里使用的是云服务器,一般云服务器默认只有动态库,如果要实现静态链接,还需安装静态库,否则会报错。我们可以通过命令:
yum install glibc-static libstdc++-static -y 来安装静态库。接着通过命令:gcc test.o -o test-static -static 实现静态链接(这里为了区分动态链接形成的可执行文件,将静态链接形成的可执行文件命名位 test-static。命令中选项 -static 表示使用静态链接和静态库)。如下图所示,可以看到静态链接形成的可执行文件的大小约有动态链接形成的可执行文件的大小的100倍,可见静态链接消耗的空间之大。

静态链接


总结: 以上为了更加详细的了解程序翻译的过程,所以分四步完成程序的翻译,在实际中,如果没有特殊需要,(以test.c文件的翻译为例)可直接使用命令 gcc test.c -o test 完成程序的编译及动态链接,使用命令 gcc test.c -o test -static 完成程序的编译及静态链接,其中如果不加选项 -o test 生成指定的目标文件,则会默认生成名为 a.out 的可执行文件。 如下图所示,无论是动态链接形成的可执行程序还是静态链接形成的可执行程序,又或是默认形成的可执行程序,最终的运行结果都是一样的。


链接


1.5 gcc常用选项总结

  • -E :只激活预处理,不生成文件,需要将执行结果重定向到一个输出文件中,否则默认将预处理结果输出到屏幕。
  • -S :编译到形成汇编语言代码就停止,不进行汇编和链接。
  • -c :编译到形成二进制目标代码。
  • -o :输出指定目标文件。
  • -static :对生成的二进制文件采用静态链接。
  • -g :生成调试信息(Debug版可执行程序)。GUN调试器可利用该信息。
  • -shared :尽量使用动态库,形成的可执行文件比较小,但前提需要系统有动态库。
  • -O0-O1-O2-O3 :编译器的优化选项的4个级别,-O0 表示没有优化,-O1 为缺省值,-O3 优化级别最高。
  • -w :不生成任何警告信息。
  • -Wall :生成所有警告信息。

以上是我对Linux中gcc编译器使用的一些学习记录总结,如有错误,希望大家帮忙指正,也欢迎大家给予建议和讨论,谢谢!


http://www.ppmy.cn/news/11437.html

相关文章

c语言操作符(下)

前言 🎈个人主页:🎈 :✨✨✨初阶牛✨✨✨ 🐻推荐专栏: 🍔🍟🌯 c语言初阶 🔑个人信条: 🌵知行合一 🍉本篇简介:>: 讲解c语言中有关操作符的知识. 金句分享: ✨✨✨行程…

Go语言结构

Go语言结构 知识主要参考菜鸟教程。 简单实例 Go语言的基础组成有以下几个部分: 包声明引入包函数变量语句 & 发表达式注释 package mainimport "fmt"func main() {/*这是一行注释*/fmt.Println("hello,world") }上述程序各个部分组成&am…

telnet Connection refused 端口不通处理

文章目录一、telnet简述二、telnet Connection refused不通处理思路排查目的主机服务一、telnet简述 telnet一般用于测试本机到目的主机的目的端口网络通不通,telnet命令后面跟目的主机的IP或者域名,格式如下: telnet ${域名}|$…

【Linux】项目自动化构建工具—make/makefile

文章目录1. 什么是make/makefile?2. make/makefile的使用2.1 实例代码2.2 依赖关系和依赖方法2.3 项目清理2.4 make是如何确定是否编译的3. Linux第一个小程序—进度条3.1 \r 和 \n3.2 进度条小程序1. 什么是make/makefile? make是一个命令工具&#xf…

WSL---Windows Subsystem for Linux

WSL:Windows Subsystem for Linux,适用于Windows的Linux子系统;是一个在Windows10/11上能够运行原生Linux二进制可执行文件(ELF格式)的兼容层。 1、介绍 1.1 文件系统 WSL支持的文件系统需要满足两个目标&#xff1…

车规级CAN FD收发器SIT1044Q,能替代TJA1044吗?

国际知名品牌NXP推出的TJA1042Q、TJA1043Q、TJA1044Q、TJA1051Q等CAN FD收发器芯片,相信很多电子工程师并不陌生。这类芯片应用中,非常成熟稳定,深受汽车电子工程师的认可、支持和青睐。然而,在实际应用中,很多客户由于…

ES6学习笔记之正则扩展

RegExp 在 ES5 中,RegExp构造函数的参数有两种情况: 一是参数是字符串,第二个参数表示正则表达式的修饰符(flag)。 var regex new RegExp(xyz, i);二是,参数是一个正则表示式,这时会返回一个原有正则表…

阿里云Docker仓库操作

为什么用阿里云docker有自己的仓库服务器,但是对没掏钱的(白嫖)用户并不是那么友好,有两个很大的限制:免费用户,在6个月之内如果没有任何操作,将会被自动删除,真的很糟糕啊。匿名用户…