既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上C C++开发知识点,真正体系化!
由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新
“-l”,
“/etc/yum.repos.d/”,
NULL
};
create_process (“ls”, arg_list); return 0;
}
**进行编译:程序的二进制格式**
## 静态链接库
* 在 Linux 下面,二进制的程序也要有严格的格式,这个格式我们称为ELF(Executeable and Linkable Format,可执行与可链接格式)。这个格式可以根据编译的结果不同,分为不同的格式。
![在这里插入图片描述](https://img-blog.csdnimg.cn/571e93fdc8304524a064ffe0eba58ebd.png)
gcc -c -fPIC process.c
gcc -c -fPIC createprocess.c
* 在编译的时候,先做预处理工作,例如将头文件嵌入到正文中,将定义的宏展开,然后就是 真正的编译过程,最终编译成为.o 文件,这就是 ELF 的第一种类型,可重定位文件 (Relocatable File)还是.o 文件,不是一个可以直接运行的程序,这里面只是 部分代码片段。
* 可重定位文件的格式 每一个格的格式叫做节section![在这里插入图片描述](https://img-blog.csdnimg.cn/84d3b3a9464241d1b9ee458542f438e9.png)
* .text:放编译好的二进制可执行代码
* .data:已经初始化好的全局变量
* .rodata:只读数据,例如字符串常量、const 的变量
* .bss:未初始化全局变量,运行时会置 0
* .symtab:符号表,记录的则是函数和变量
* .strtab:字符串表、字符串常量和变量名
* .rel.text, .rel.data 就与重定位有关 , 存放被调用函数的位置
* `create_process` 这个函数作为库文件被重用,不能以.o 的形式存在,而是要形成库 文件,最简单的类型是静态链接库.a 文件(Archives),仅仅将一系列对象文件(.o)归档 为一个文件,使用命令 ar 创建。
`ar cr libstaticprocess.a process.o # libstaticprocess.a 里面只有一个.o,实际可以有多个`
* `gcc -o staticcreateprocess createprocess.o -L. -lstaticprocess`
-L 表示在当前目录下找.a 文件,-lstaticprocess 会自动补全文件名,比如 加前缀 lib,后缀.a,变成 libstaticprocess.a,找到这个.a 文件后,将里面的 process.o 取 出来,和 createprocess.o 做一个链接,形成二进制执行文件 staticcreateprocess。
`运行 # ./staticcreateprocess`
二进制文件格式
![在这里插入图片描述](https://img-blog.csdnimg.cn/2f6b6f9ff6de485cb63df635616afff6.png)
>
> 这个section 是多个.o 文件合并过的。
> 因而这些 section 被分成了需要加载到内存里面的代码段、数据段和不 需要加载到内存里面的部分,将小的 section 合成了大的段 segment,并且在最前面加一 个段头表(Segment Header Table)。在代码里面的定义为 struct elf32\_phdr 和 struct elf64\_phdr,(下面有)这里面除了有对于段的描述之外,最重要的是 p\_vaddr,这个是这个段加载 到内存的虚拟地址。
>
>
>
## 动态链接库
对于静态链接库, 缺点就是相同的代码段,如果被多个程序使用的 话,在内存里面就有多份,而且一旦静态链接库更新了,如果二进制执行文件不重新编译, 也不随着更新。
gcc -shared -fPIC -o libdynamicprocess.so process.o # 多了 -shared -fPIC
gcc -o dynamiccreateprocess createprocess.o -L. -ldynamicprocess
export LD_LIBRARY_PATH=. # 首先寻找动态链接库,然后加载它
./dynamiccreateprocess
* 动态链接库,就是 ELF 的第三种类型,共享对象文件(Shared Object)。
## 运行程序为进程
内核中,有这样一个数据结构,用来定义加载二进制文件的方法。
struct linux_binfmt {
struct list_head lh; struct module *module;
int (*load_binary)(struct linux_binprm *); int (*load_shlib)(struct file *); int (*core_dump)(struct coredump_params *cprm); unsigned long min_coredump;
/* minimal dump size */ } __randomize_layout;
对于 ELF 文件格式,有对应的实现。
static struct linux_binfmt elf_format = {
.module= THIS_MODULE,
.load_binary .load_shlib .core_dump
= load_elf_binary, = load_elf_library, = elf_core_dump,
.min_coredump = ELF_EXEC_PAGESIZE,
};
exec 比较特殊,它是一组函数:
包含 p 的函数(execvp, execlp)会在 PATH 路径下面寻找程序;
不包含 p 的函数需要输入程序的全路径;
包含 v 的函数(execv, execvp, execve)以数组的形式接收参数;
包含 l 的函数(execl, execlp, execle)以列表的形式接收参数;
包含 e 的函数(execve, execle)以数组的形式接收环境变量。
process.c 的代码中,我们创建 ls 进程,也是通过 exec。
## 进程树
![img](https://img-blog.csdnimg.cn/img_convert/326f4f8f953e6cb91913c424e9ca6402.png)
![img](https://img-blog.csdnimg.cn/img_convert/f9ba29791b76a8ac6d5fe965be05ea1e.png)
**网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。**
**[需要这份系统化的资料的朋友,可以添加戳这里获取](https://bbs.csdn.net/topics/618668825)**
**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**
n.net/topics/618668825)**
**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**