『Linux - gcc / g++』c程序翻译过程

简介: 『Linux - gcc / g++』c程序翻译过程

前言

在计算机中的每一个程序是由代码变化而来的,但是事实上来说,用 c/C++ 写出的代码是不能被计算机识别的,其中必须经过一系列的过程才能使这个代码能成功的被计算机识别;

这几个过程分别为:

预处理 编译 汇编 链接

在经历过这几个过程之后才能将代码转换为一个可执行程序;


预处理 -E

在预处理阶段时一般会进行:

  • 去注释 - 将源文件中的注释使用"空格"代替;
  • 宏替换 - 若是源文件中存在以#define定义的宏,则将其替换;
  • 头文件展开 - 将头文件进行展开;
  • 条件编译 - 根据条件编译中的条件,留下正确条件;

该些操作一般为文本操作,且该些操作结束后也仍为c/C++;

g++ -E test.cpp test.cpp文件进行预处理,并将预处理后的结果显示在显示器上;
g++ -E test.cpp -o test.i test.cpp文件进行预处理,同时指定生成一个 test.i临时文件;
-E 指令 从现在开始进行程序的翻译,如果预处理完成则停止;

从上图可以看出预处理前后的区别

可以看出两个文件大小的差距,源文件中的代码量大概只有23行,而再进行预处理过后,代码量变成接近三万行;


编译 -S

编译这个过程则是将预处理后的 .i 文件进行翻译,将c/C++代码翻译成汇编代码 .s文件;

g++ -S test.i 或是 g++ -S test.i -o test.s test.i文件编译成汇编语言 , 两种方法都可以生成test.s文件 ; 第二种方法可以指定生成文件的文件名(这里后缀一般不影响);
g++ -S test.cpp -o test.s 作用同上,唯一不同的为这里将从test.cpp文件开始进行翻译,途中将要重新进行一次预处理;;
-E 指令 从现在开始进行程序的翻译直到编译完成;


汇编 -c

在这个阶段过后,.s文件将会被转为可重定向二进制目标文件( .o 文件);

虽然这里转化为了二进制文件,但是仍然不能被计算机直接进行识别;

g++ -c test.s 或是 g++ -c test.i -o test.o 将汇编语言文件 test.s 进行汇编生成可重定向二进制目标文件 test.o
od 命令 将指定内容文件以八进制、十进制、十六进制、浮点格式或ASCII编码字符方式显示。默认使用的是八进制;
-c 指令 从现在开始进行程序翻译,直到汇编结束。

[如图所示为使用vim打开时所显示的为二进制]

这里所生成的可重定向二进制文件并不能被执行;

这里或多或少可能会存在权限问题,因为所生成的可重定向二进制文件的权限为666 & (~umask)为664;

即没有运行的权限,为了排除该情况,修改文件权限;

但即使修改了文件权限也依然不能运行;

需要打开该文件即可以使用od命令将文件打开;


链接

这也是程序翻译的最后一个步骤,即将多个.o 或者 .obj 文件合并成一个可执行程序 .exe ;

g++ test.cpp -o test 或 g++ -o testtest.cpp test.cpp源文件编译生成 test可执行程序;

然而在链接这一过程中也有分情况,分别为动态链接与静态链接;


动静态链接

在这之前首先要介绍两个命令:

ldd filename file filename
查看该文件所依赖的库文件 可以查看文件的可执行程序状态(位数、动态链接等)

一般的连接方式分为两种:

a.动态链接 b.静态链接
需要动态库 需要静态库

在调用函数中总是会包含各种头文件,但只有头文件是并不能将程序编译通过的;

头文件中只包含了各个函数的声明,而函数的定义一般以库的形式展示;

步骤一般为,在调用函数过后,从头文件中找到函数的声明,再去对应的库中找到函数的定义从而进行调用;

头文件 : 用来提供函数的声明 库文件 : 用来提供函数的定义(实现)

自己写的源文件包含头文件,链接库文件才能生成一个可执行程序;

系统\库 动态库 静态库
Linux .so .a
Windows .dll .lib

在安装vs2019或者vs2022这种编译器时,虽然叫做配置环境,但是本质上是在安装所用语言的头文件以及库文件;

C程序是脱离不开库文件的;

在Linux中,许多的命令也是利用c语言实现的,例如在Linux中使用的ls命令;

在汇编过后所生成的可重定向二进制目标文件;

#include<iostream>
int main()
{
  std::cout << "it's a test fail" << std:: endl;
  return 0;
}

从该文件中可以看到,该文件包含了一个文件 ,且调用了流插入<<流提取>>;

而这里生成的可重定向二进制文件只是将自己写的源文件中的代码生成可重定向的二进制目标文件;

至于代码中所包含的头文件以及使用流插入流提取并没有进行操作;

该操作将会在最后一步的链接过程中;

链接过程将会把该重定向二进制文件与对应的库文件进行链接最终生成一个可执行程序;(头文件在预处理阶段被展开)

静态链接 动态链接
原理 在生成可执行文件的时候(链接阶段),把所有需要的函数的二进制代码都包含到可执行文件中去。因此,链接器需要知道参与链接的目标文件需要哪些函数,同时也要知道每个目标文件都能提供什么函数,这样链接器才能知道是不是每个目标文件所需要的函数都能正确地链接。如果某个目标文件需要的函数在参与链接的目标文件中找不到的话,链接器就报错了。目标文件中有两个重要的接口来提供这些信息:一个是符号表,另外一个是重定位表。 在编译的时候不直接拷贝可执行代码,而是通过记录一系列符号和参数,在程序运行或加载时将这些信息传递给操作系统,操作系统负责将需要的动态库加载到内存中,然后程序在运行到指定的代码时,去共享执行内存中已经加载的动态库可执行代码,最终达到运行时连接的目的。
缺点 静态库的缺点为,若是多个文件同时调用同一个函数,则可能出现在一个项目中出现大量重复代码; 动态库链接的缺点即为过于依赖动态库;
方法 g++ test.cpp -o test -static
(-static : 表明使用静态链接的方式形成可执行程序)
g++ test.cpp -o test (默认即为动态链接)

  1. 一般的机器可能会因为没有静态库而导致静态链接失败;
    在Linux中,动态链接必须使用 .so 动态库文件;
    静态链接必须使用 .a 静态库文件;
  2. 若是没有静态库文件则需要进行安装 :
c静态库安装命令 C++静态库安装命令
(sudo yum install -y glibc-static) (sudo yum install -y libstdc+±static)
相关文章
|
8天前
|
Linux 开发工具 C语言
Linux 安装 gcc 编译运行 C程序
Linux 安装 gcc 编译运行 C程序
31 0
|
8天前
|
Linux Android开发
测试程序之提供ioctl函数应用操作GPIO适用于Linux/Android
测试程序之提供ioctl函数应用操作GPIO适用于Linux/Android
12 0
|
7天前
|
Java Shell Linux
【linux进程控制(三)】进程程序替换--如何自己实现一个bash解释器?
【linux进程控制(三)】进程程序替换--如何自己实现一个bash解释器?
|
7天前
|
Shell Linux 程序员
【linux进程(六)】环境变量再理解&程序地址空间初认识
【linux进程(六)】环境变量再理解&程序地址空间初认识
|
29天前
|
监控 Java Linux
使用jvisualVM监控远程linux服务器上运行的jar程序
使用jvisualVM监控远程linux服务器上运行的jar程序
13 5
|
2月前
|
Linux Shell
【Linux】进程与可执行程序的关系&&fork创建子进程&&写实拷贝的理解
【Linux】进程与可执行程序的关系&&fork创建子进程&&写实拷贝的理解
|
2月前
|
算法 Linux 编译器
⭐⭐⭐⭐⭐Linux C++性能优化秘籍:从编译器到代码,探究高性能C++程序的实现之道
⭐⭐⭐⭐⭐Linux C++性能优化秘籍:从编译器到代码,探究高性能C++程序的实现之道
160 2
|
Linux C语言 C++
在64位linux下编译32位程序
一般情况下我们最好是在一个平台上编译本平台的二进制程序,比如在32位平台上编译32位程序而在64位平台上编译64位程序。    现在64位的系统(这里主要指x86_64系统,包括AMD64和Intel的EM64T)已经开始广泛的使用起来了,甚至于你现在要是想买32位的处理器根本就没有。
2004 0
|
2天前
|
监控 Linux Perl
【专栏】Linux 命令小技巧:显示文件指定行的内容
【4月更文挑战第28天】本文介绍了Linux中显示文件指定行内容的方法,包括使用`head`和`tail`命令显示文件头尾部分,利用`sed`的行号指定功能以及`awk`处理文本数据。文章还列举了在代码审查、日志分析和文本处理中的应用场景,并提醒注意文件编码、行号准确性及命令组合使用。通过练习和实践,可以提升Linux文本文件处理的效率。
|
2天前
|
运维 网络协议 Linux
【专栏】运维工程师工作时最常用的 20 个 Linux 命令有哪些?建议收藏
【4月更文挑战第28天】本文介绍了运维工程师常用的20个Linux命令,包括`ls`、`cd`、`pwd`、`mkdir`、`rm`、`cp`、`mv`、`cat`、`more`、`less`、`head`、`tail`、`grep`、`find`、`chmod`、`chown`、`chgrp`、`ps`、`top`和`ifconfig`,帮助提升工作效率。此外,还提到了其他常用的命令如`df`、`free`、`tar`、`ssh`、`scp`、`ping`、`netstat`、`iptables`、`systemctl`、`hostname`等,建议运维人员掌握以应对各种运维场景。