2024年最新【Linux操作系统】进程_将动态链接库启动为进程,教你一种更清晰的C C++架构

img
img

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

extern int create_process (char* program, char** arg_list);
int create_process (char* program, char** arg_list) {
pid_t child_pid;
child_pid = fork (); //fork 系统调用
if (child_pid != 0) return child_pid;
else {
execvp (program, arg_list); // execvp 运行一个新的程
abort ();
}
return 0;
}


`createprocess.c`



#include <stdlib.h>
#include <sys/types.h>
#include <unistd.h>
extern int create_process (char* program, char** arg_list); //创建子进程运行 命令 ls
int main () {
char* arg_list[] = {
“ls”,
“-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;




![img](https://img-blog.csdnimg.cn/img_convert/05051ecb909dff68febada2132fbbc74.png)
![img](https://img-blog.csdnimg.cn/img_convert/82cb2913153dc7c9916e43e475df1a99.png)

**网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。**

**[需要这份系统化的资料的朋友,可以添加戳这里获取](https://bbs.csdn.net/topics/618668825)**


**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**

dn.net/topics/618668825)**


**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**

  • 5
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值