编译和链接那点事感悟 -2018.9.3

在拜读了刘欢所写的编译和链接那点事后,对编译和链接有了深一步的理解,所谓编译链接即是将无法直接运行的高级语言等转化为计算机可以直接识别的机器语言必不可少的步骤,根据流程画与理解出了gcc编译器的四个过程以及所做工作:
gcc全过程
gcc分为四个步骤:预处理(cpp)、编译(cc1)、汇编(as)、链接(ld)。

  • 预处理:头文件展开,宏定义展开,注释替换等,这里的展开是将宏定义所有的展开。
extern char *ctermid (char *__s) __attribute__ ((__nothrow__ , __leaf__));
# 840 "/usr/include/stdio.h" 3 4
extern void flockfile (FILE *__stream) __attribute__ ((__nothrow__ , __leaf__));



extern int ftrylockfile (FILE *__stream) __attribute__ ((__nothrow__ , __leaf__)) ;


extern void funlockfile (FILE *__stream) __attribute__ ((__nothrow__ , __leaf__));
# 868 "/usr/include/stdio.h" 3 4

# 2 "Helloworld.c" 2


# 3 "Helloworld.c"
void main(void)
{
 printf("helloworld!\n");
}

宏定义为#include”stdio.h”的Helloworld.c预处理(部分)

# 1 "Helloworld.c"
# 1 "<built-in>"
# 1 "<command-line>"
# 31 "<command-line>"
# 1 "/usr/include/stdc-predef.h" 1 3 4
# 32 "<command-line>" 2
# 1 "Helloworld.c"


void main(void)
{
int printf(const char *format,...);
 printf("helloworld!\n");
}

只写入printf函数的Helloworld.c预处理(全部)
以上对比可看到,尽管不需要使用除printf之外其他函数的定义,却还是依旧展开了全部,同时只对printf定义也成功通过了编译,在平时我们在用gcc编译时是否是第一步出问题时,可以考虑使用gcc -E来检查。

  • 编译:将预处理文件进行词法分析、语法分析生成汇编代码文件,这是gcc最核心同时也是最复杂的部分。但此处具体较为复杂,同时需要大量汇编知识,因此笔者并未展开详细说,但是我们可以把他理解为降阶,从高级语言降阶为汇编代码。具体操作为gcc -S,生成后缀为.s的文件,打开为汇编代码。
  • 汇编:将翻译后的汇编代码翻译为机器码,几乎每一条汇编指令对应一句机器码。把汇编代码生成后缀为.o的目标文件,目标文件就是经过编译但未进行连接的中间文件,Linux下.o就是目标文件,在Linux下我们可以将目标文件和可执行文件看做同一类型,他们都以ELF文件格式存储。(Linux下的ELF文件类型包括.o文件,可执行文件、核心转储文件(core dump)以及so文件(动态链接库))。
  • 最后的步骤则是链接,笔者在这里做了非常详细的说明。何为链接?我们已经将我们所熟悉的高级语言转为了机器码,机器码距离我们最终可执行文件已经非常接近了,但机器码在执行的时候需要很重要的一点就是支撑它执行的库,例如我们常见的C标准库等,链接即把机器码和这些库连接起来,如同把你看书碰到不会的字需要去查字典一样,库分为静态库与动态库。
add.c  add.o  calc.h  libcalc.a  swap.c  swap.o  test  test.c
hawl29@hawl29-PC:~/Desktop/test$ rm add.o
hawl29@hawl29-PC:~/Desktop/test$ rm swap.o
hawl29@hawl29-PC:~/Desktop/test$ ls
add.c  calc.h  libcalc.a  swap.c  test  test.c
hawl29@hawl29-PC:~/Desktop/test$ rm libcalc.a 
hawl29@hawl29-PC:~/Desktop/test$ ls
add.c  calc.h  swap.c  test  test.c
hawl29@hawl29-PC:~/Desktop/test$ rm test
hawl29@hawl29-PC:~/Desktop/test$ ls
add.c  calc.h  swap.c  test.c
hawl29@hawl29-PC:~/Desktop/test$ gcc add.c swap.c  -shared -o libcalc.so
hawl29@hawl29-PC:~/Desktop/test$ ls
add.c  calc.h  libcalc.so  swap.c  test.c
hawl29@hawl29-PC:~/Desktop/test$ gcc test.c  -o test ./libcalc.so 
hawl29@hawl29-PC:~/Desktop/test$ ./test 
2 1
hawl29@hawl29-PC:~/Desktop/test$ ldd ./test
    linux-vdso.so.1 (0x00007ffd84feb000)
    ./libcalc.so (0x00007f0743b96000)
    libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f07437a5000)
    /lib64/ld-linux-x86-64.so.2 (0x00007f0743f9a000)
hawl29@hawl29-PC:~/Desktop/test$ 

分别感受静态库和动态库的区别
了解到:

  • 静态库即把库中所有文件编译,需要用哪个的时候直接链接与需要的那部分链接,这样避免了资源浪费,但同时也有很大的缺点就是当库更新后所有的库函数必须重新编译,以及每个程序都要和许多相关文件进行链接,这样很浪费空间。由此产生了动态库。
  • 动态库如同一个可释放的静态库,当只有在运行时,由动态链接器完成与程序的链接,这样极大程度的解决了存储空间浪费的问题。
  • 链接具体分为空间地址分配、符号决议和重定位等,体现为先找出分配存放它的地址空间,接着统计所有函数等编译后留下的符号,相同同的进行强弱对比,接着寻找每一个确定的符号位置,修正每一个符号地址,最终完成确定的链接。
版权声明:本文为weixin_43122409原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/weixin_43122409/article/details/82384143

智能推荐

Java HashMap那点事

集合类的整体架构 比较重要的集合类图如下: HashMap详解    HashMap 和 HashSet 是 Java Collection Framework 的两个重要成员,其中 HashMap 是 Map 接口的常用实现类,HashSet 是 Set 接口的常用实现类。虽然 HashMap 和 HashSet 实现的接口规范不同,但它们底层的 Hash 存储机制完全一样,甚至 HashSet...

fork那点事

fork那点事 fork 总结 fork()通过复制调用进程来创建一个新进程。在Linux下,fork()是通过使用写时复制页面实现的,所以它唯一的缺点是复制父页表的时间和内存,并为子进程创建独特的任务结构。 fork调用的一个奇妙之处就是它仅仅被调用一次,却能够返回两次,它有三种不同的返回值:     1)在父进程中,fork返回新创建子进程的进程ID;  ...

信号那点事(一)

信号 概念      信号是软件中断。很多⽐较重要的应⽤程序都需处理信号。信号提供了⼀种处理异步事件的⽅法。每个信号都有⼀个名字。这些名字都以三个字符SIG开头。   查看系统支持的信号     在Linux系统中可以使用kill -l命令来查看系统支持的信号。   信号发生的条件 1) ...

ES那点事

1.关于大数据导入es的问题,,, sql批量导入,记录保留位置     关于es的内存方案       转载https://blog.csdn.net/liuming690452074/article/details/85042813  ...

关于记账那点事

记录一些关于记账的Tips 1.通过记账实现“优化财务状况”的目的 -“坚持”记账 习惯回路:由线索(在什么环境或场景下,使用这个习惯)、行为(习惯的具体内容和操作步骤)、奖赏(做完之后,如何复盘并得到反馈)三元素组成 -复盘很重要 通过记账了解自己的“消费黑洞”,并以此作为依据进行复盘,从何制定更为合理的消费预算、储蓄计划;...

猜你喜欢

构造方法那点事!

Java构造方法测试 关于父类的构造方法结论: 1.调用那个构造方法使用那个构造方法 2.用对象初始化对象的时候Java不会调用拷贝构造方法,而是采用了一种引用方法,给father1起了了名字叫father3,没有新的对象生成。 关于子类的构造方法结论: 1.默认调用父类的构造方法,会父类同名参数和,同名函数。可以通过super来进行对父类的内容访问。 2.用对象初始化对象的时候Java不会调用拷...

Linux环境下配置和安装hadoop及hadoop集群搭建(VMware)

文章目录 一、安装准备 二、hadoop的配置 1.首先配置hadoop-env.sh 2.配置core-site.xml 3.配置hdfs-site.xml 4.配置mapred-site.xml 5.配置yarn-site.xml 6.配置slaves 7.配置hadoop环境变量 三、格式化HDFS 四、启动hadoop 五、集群搭建 1.克隆虚拟机 2.配置免密登录 3.修改主机器的配置文...

使用QProcess打开和关闭第三方应用,比如CMD

使用QProcess打开和关闭第三方应用,比如CMD 注意: 很多教程不一定是对的,但我这篇绝对是对的,因为我踩坑过啊。 为了节省时间,直接上图、上代码,so easy! 重要事情说3遍: 杀死进程,一定要加/F 和 /T 杀死进程,一定要加/F 和 /T 杀死进程,一定要加/F 和 /T 开始 验证下,打开任务管理器就能看到 总结 从上面看,是不是很简单,taskkill不知道是啥,是windo...

自定义View实现注销图案的加载动画

先看效果图: 有那味了。。。(懂得都懂^ ^ √) 我们先来分析一下怎么画,然后再研究怎么让他动起来 这个View是由内部的注销图案和外面一圈圆环构成。而内部的注销图案又是由一个基本满角度的圆弧和一根竖线组成 一、绘制内部注销图案 首先初始化画笔和圆弧的外切矩形: 圆弧的中心是View的中心,坐标为(getWidth()/2,getWidth()/2),半径设置为getWidth/4,...

vue3使用vue-count-to组件

项目场景: 数据可视化大屏开发的过程中,需要实现一种滚动数字的效果,在使用vue2时,使用vue-count-to完全没有问题,功能也比较完善(滚动时长,开始值,结束值,前缀,后缀,千分隔符,小数分隔符等等),但是在vue3中使用会出现问题。 展示的效果 问题描述: 出现的错误时 == Cannot read property ‘_c’ of undefined== 这是一...