最新资讯

  • 【Linux】ELF与动静态库的“暗黑兵法”:程序是如何跑起来的?

【Linux】ELF与动静态库的“暗黑兵法”:程序是如何跑起来的?

2025-05-20 07:38:02 0 阅读

目录

一、什么是库?

 1. C标准库(libc)

2. C++标准库(libstdc++)

 二、静态库

1. 静态库的生成

2. 静态库的使用

三、动态库

1. 动态库的生成

2. 动态库的使用

3. 库运行的搜索路径。

(1)原因分析

(2)解决方案

① 设置 LD_LIBRARY_PATH

② 将库复制到系统路径

③ 复制.so文件到系统库目录

④ 创建软链接到系统库文件

四、外部库(补充)

五、目标文件(原理部分)

六、EIL文件

1. ELF文件的四种格式

2. ELF文件的核心结构

(1)ELF Header(ELF头)

(2)Program Header Table(程序头表)

(3)Section Header Table(节头表)

(4)Sections(节)

七、ELF从加载到轮廓

1. ELF形成可执行文件

2. ELF可执行文件加载

八、理解链接与加载

1. 静态链接

(1)编译阶段

(2)重定位表

 (3)静态链接阶段:地址修正

(4)总结

2. ELF加载和进程地址空间

(1)逻辑地址 / 虚拟地址 

(2)重新理解进程虚拟地址空间

(3)静态链接库在内存加载 

3. 动态链接与动态库加载

(1)动态库加载

(2)进程间共享动态库

(3)动态链接

① 动态链接概要

② 可执行程序被编译器动了手脚

③ 动态库中的相对地址

④ 程序与动态库映射

⑤ 程序调用库函数

⑥ 全局偏移量表GOT 

⑦ 动态库间依赖

4. 总结


一、什么是库?

库是写好的现有的,成熟的,可以复用的代码。现实中每个程序都要依赖很多基础的底层库,不可能每个人的代码都从零开始,因此库的存在意义非同寻常。

本质上来说库是一种可执行代码的二进制形式,可以被操作系统载入内存执行。

库有两种:静态库.a[Linux]、.lib[windows];动态库.so[Linux]、.dll[windows]

 1. C标准库(libc)

系统动态库(.so)静态库(.a)
Ubuntu/lib/x86_64-linux-gnu/libc-2.31.so/lib/x86_64-linux-gnu/libc.a
-rwxr-xr-x 1 root root 2029592 May 1 02:20-rw-r--r-- 1 root root 5747594 May 1 02:20
CentOS/lib64/libc-2.17.so/lib64/libc.a
-rwxr-xr-x 1 root root 2156592 Jun 4 23:05-rw-r--r-- 1 root root 5105516 Jun 4 23

2. C++标准库(libstdc++)

系统动态库(.so)静态库(.a)
Ubuntu/usr/lib/gcc/x86_64-linux-gnu/9/libstdc++.so/usr/lib/gcc/x86_64-linux-gnu/9/libstdc++.a
符号链接→ ../../../x86_64-linux-gnu/libstdc++.so.6-rw-r--r--(文件大小未显示)
CentOS/lib64/libstdc++.so.6/usr/lib/gcc/x86_64-redhat-linux/4.8.2/libstdc++.a
符号链接→ libstdc++.so.6.0.19-rw-r--r-- 1 root root 2932366 Sep 30 2020

 二、静态库

• 静态库(.a):程序在编译链接的时候把库的代码链接到可执行文件中,程序运行的时候将不再需要静态库。

• 一个可执行程序可能用到许多的库,这些库运行有的是静态库,有的是动态库,而我们的编译默认为动态链接库,只有在该库下找不到动态.so的时候才会采用同名静态库。我们也可以使用 gcc 的 -static 强制设置链接静态库,一旦 -static 就必须存在对应的静态库。

1. 静态库的生成

mystdio.c mymakstring.c是我们自主实现的源文件。现将其编译为目标文件(.o),再使用ar打包为静态库(libmystdio.a)。

output:将头文件(.h)和静态库(.a)打包成 stdc.tgz。步骤:

(1)创建目录 stdc/include 和 stdc/lib。
(2)复制头文件到 include/,静态库到 lib/。

(3)用 tar -czf 打包成 stdc.tgz(gzip 压缩)。

ar 是归档工具,参数说明:
r:替换已存在的文件
c:创建库(如果不存在)
s:写入索引(加速链接)
t 选项:列出静态库中的文件
v 选项:详细信息(verbose)

[Makefile]
libmystdio.a:mystdio.o mymakstring.o
	@ar -rc $@ $^
	@echo "build $^ to $@ ... done"
%.o:%.c
	@gcc -c $<
	@echo "compling $< to $@ ... done"
.PHONY:clean
clean:
	@rm -rf *.a *.o stdc*
	@echo "clean ... done"
.PHONY:output
output:
	@mkdir -p stdc/include
	@mkdir -p stdc/lib
	@cp -f *.h stdc/include
	@cp -f *.a stdc/lib
	@tar -czf stdc.tgz stdc
	@echo "output stdc ... done"

2. 静态库的使用

任意目录下新建Main.c,使用我们自己实现的库里的函数调用。

// 再新建目录lib下的新文件Main.c
#include "mystdio.h"
#include "mystring.h"

int main()
{
    MYFILE *filep = MyFopen("log.txt", "a");
    if (!filep)
    {
        printf("MyFopen error!
");
        return 1;
    }

    // const char *msg = "hello MyFwrite
"; // 行刷新
    // MyFwrite(filep, msg, strlen(msg));

    int cnt = 5;
    while (cnt--)
    {
        const char *msg = "hello MyFwrite!"; // 没有'
',不满足刷新条件,待在缓冲区
        MyFwrite(filep, msg, strlen(msg));
        // 强制刷新缓冲区
        MyFflush(filep);
        printf("buffer:%s
", filep->outbuffer); // 打印缓冲区内容
        sleep(1);
    }

    MyFcolse(filep);
    const char *str = "hello!
";
    printf("my_strlen: %d
", my_strlen(str));
    return 0;
}

场景1:头文件和库文件安装到系统路径下
$ gcc Main.c -lmystdio

场景2:头文件和库文件和我们自己的源文件在同一个路径下
$ gcc Main.c -L. -lmystdio

场景3:头文件和库文件有自己的独立路径
$ gcc Main.c -I头文件路径 -L库文件路径 -lmystdio

-L:指定库路径。
-I:指定头文件搜索路径。
-l:指定库名。

• 测试目标文件生成后,静态库删掉,程序照样可以运行。
• 关于 -static 选项,稍后介绍。
库文件名称和引入库的名称:去掉前缀 lib,去掉后缀.so、.a,如:libc.so -> c

$ tree .
.
├── lib
│   └── Main.c
├── Makefile
├── mystdio.c
├── mystdio.h
├── mystring.c
├── mystring.h
└── usercode.c
2 directories, 10 files

$ make   # 制作并打包静态库
compling mystdio.c to mystdio.o ... done
compling mystring.c to mystring.o ... done
build mystdio.o mystring.o to libmystdio.a ... done

$ cd lib
$ gcc Main.c -I../ -L../ -lmystdio # 链接静态库
$ ./a.out
buffer:hello MyFwrite!
buffer:hello MyFwrite!
buffer:hello MyFwrite!
^C

$ tree ../
../
├── lib
│   ├── a.out
│   └── Main.c
├── libmystdio.a
├── Makefile
├── mystdio.c
├── mystdio.h
├── mystdio.o
├── mystring.c
├── mystring.h
├── mystring.o
└── usercode.c

2 directories, 11 files

三、动态库

• 动态库(.so):程序在运行的时候才去链接动态库的代码,多个程序共享使用库的代码。

• 一个与动态库链接的可执行文件仅仅包含它用到的函数入口地址的一个表,而不是外部函数所在目标文件的整个机器码。

在可执行文件开始运行以前,外部函数的机器码由操作系统从磁盘上的该动态库中复制到内存中,这个过程称为动态链接(dynamic linking)。

• 动态库可以在多个程序间共享,所以动态链接使得可执行文件更小,节省了磁盘空间。操作系统采用虚拟内存机制允许物理内存中的一份动态库被要用到该库的所有进程共用,节省了内存和磁盘空间。

1. 动态库的生成

(1)将 mystdio.o 和 mystring.o 链接成动态库 libmystdio.so。

-shared:生成动态库/共享库格式(而不是可执行文件)。
库名规则:libxxx.so

(2)将 .c 文件编译为位置无关代码(PIC)的目标文件(.o)。

-fPIC:生成位置无关代码(Position-Independent Code),动态库必需。
$<:当前依赖的源文件(如 mystdio.c)。

(3)将头文件(.h)和动态库(.so)打包成 stdc.tgz。

[Makefile]
libmystdio.so:mystdio.o mystring.o
	@gcc -o $@ $^ -shared
	@echo "build $^ to $@ ... done"
%.o:%.c
	@gcc -fPIC -c $< 
	@echo "compling $< to $@ ... done"
.PHONY:clean
clean:
	@rm -rf *.so *.o stdc*
	@echo "clean ... done"
.PHONY:output
output: # 把头文件和动态库打包压缩成 stdc.tgz
	@mkdir -p stdc/include
	@mkdir -p stdc/lib
	@cp -f *.h stdc/include
	@cp -f *.so stdc/lib
	@tar -czf stdc.tgz stdc
	@echo "output stdc ... done"

2. 动态库的使用

场景1:头文件和库文件安装到系统路径下
$ gcc main.c -lmystdio

场景2:头文件和库文件和我们自己的源文件在同一个路径下
$ gcc main.c -L. -lmymath // 从左到右搜索-L指定的目录

场景3:头文件和库文件有自己的独立路径
$ gcc main.c -I头文件路径 -L库文件路径 -lmymath

$ gcc usercode.c -I../stdc/include -L../stdc/lib -lmystdio
$ ll
total 28
drwxrwxr-x 2 zyt zyt  4096 May 14 18:47 ./
drwxrwxr-x 4 zyt zyt  4096 May 14 18:45 ../
-rwxrwxr-x 1 zyt zyt 16256 May 14 18:47 a.out*
-rw-rw-r-- 1 zyt zyt   742 May 14 18:29 usercode.c
# 当我们执行代码时,却显示动态库没有被加载
$ ./a.out
./a.out: error while loading shared libraries: libmystdio.so: cannot open shared object file: No such file or directory
# 用ldd查看库或可执行程序的依赖,发现libstdio.so找不到
$ ldd a.out
        linux-vdso.so.1 (0x00007ffe3f0f1000)
        libmystdio.so => not found
        libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x0000769ff5a00000)
        /lib64/ld-linux-x86-64.so.2 (0x0000769ff5cf4000)

我们按照方法执行后发现,动态库没有被加载。 这是为什么?

这个问题是因为系统在运行时找不到动态库 libmystdio.so 的位置。虽然我们在编译时通过 -L 指定了库的路径,但 -L 只对编译时的链接器有效,而运行时的动态链接器(ld.so)并不知道这个路径。

3. 库运行的搜索路径。

(1)原因分析

对于前面的问题:

$ ldd a.out
        linux-vdso.so.1 (0x00007ffe3f0f1000)
        libmystdio.so => not found
        libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x0000769ff5a00000)
        /lib64/ld-linux-x86-64.so.2 (0x0000769ff5cf4000)

• 编译时:-L../stdc/lib 告诉链接器在哪里找 libmystdio.so,因此编译能成功。

• 运行时:系统默认只在标准路径(如 /lib、/usr/lib)和 LD_LIBRARY_PATH(环境变量)中搜索动态库,而你的库在自定义路径 ../stdc/lib 中,导致加载失败。

(2)解决方案

① 设置 LD_LIBRARY_PATH

LD_LIBRARY_PATH是环境变量。作用是将库所在路径添加到动态链接器的搜索路径中。

缺点:仅在当前终端会话有效,重启后需重新设置。

$ echo $LD_LIBRARY_PATH # 初始是空

$ export LD_LIBRARY_PATH=${LD_LIBRARY_PATH}:/home/zyt/linux-journey-log/code_25_5_14/dynamiclib/stdc/lib
$ echo ${LD_LIBRARY_PATH}
:/home/zyt/linux-journey-log/code_25_5_14/dynamiclib/stdc/lib
$ ldd a.out
linux-vdso.so.1 (0x00007ffd45df0000)
libmystdio.so => /home/zyt/linux-journey-log/code_25_5_14/dynamiclib/stdc/lib/libmystdio.so (0x00007f8a1b234000)
libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f8a1ae00000)
/lib64/ld-linux-x86-64.so.2 (0x00007f8a1b434000)
② 将库复制到系统路径

ldconfig方案:配置/etc/ld.so.conf.d/ 。该目录包含自定义的库路径配置文件,系统启动时会加载这些路径到动态链接器的缓存中。

# /etc/ld.so.conf.d/下创建一个自定义的配置文件
$ sudo touch /etc/ld.so.conf.d/zyt.conf 
[sudo] password for zyt: 
$ ll /etc/ld.so.conf.d
total 36
drwxr-xr-x   2 root root  4096 May 14 19:10 ./
drwxr-xr-x 122 root root 12288 May  8 06:37 ../
-rw-r--r--   1 root root    38 Jan 22  2024 fakeroot-x86_64-linux-gnu.conf
-rw-r--r--   1 root root    44 Aug  2  2022 libc.conf
-rw-r--r--   1 root root   100 Mar 30  2024 x86_64-linux-gnu.conf
-rw-r--r--   1 root root     0 May 14 19:10 zyt.conf
-rw-r--r--   1 root root    56 Jan 29 01:07 zz_i386-biarch-compat.conf
-rw-r--r--   1 root root    58 Jan 29 01:07 zz_x32-biarch-compat.conf

# 填写动态库路径
$ sudo vim /etc/ld.so.conf.d/zyt.conf
$ cat /etc/ld.so.conf.d/zyt.conf
/home/zyt/linux-journey-log/code_25_5_14/dynamiclib/stdc/lib

# 更新库缓存,重新加载库搜索路径,使生效
$ sudo ldconfig
$ ldd a.out
        linux-vdso.so.1 (0x00007ffc84daa000)
        libmystdio.so => /home/zyt/linux-journey-log/code_25_5_14/dynamiclib/stdc/lib/libmystdio.so (0x000071e2461f0000)
        libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x000071e245e00000)
        /lib64/ld-linux-x86-64.so.2 (0x000071e2461fc000)
③ 复制.so文件到系统库目录

拷贝.so文件到系统共享库路径下,一般是/usr/lib,/usr/local/lib,/lib64。(推荐)

# 1. 复制动态库到/usr/local/lib(需要sudo权限)
$ sudo cp /home/zyt/linux-journey-log/code_25_5_15/dynamiclib/stdc/lib/libmystdio.so /usr/local/lib/
# 2. 设置正确的文件权限
$ sudo chmod 755 /usr/local/lib/libmystdio.so
# 3. 更新动态链接器缓存
$ sudo ldconfig
# 4. 验证是否成功
$ ldconfig -p | grep libmystdio.so
        libmystdio.so (libc6,x86-64) => /usr/local/lib/libmystdio.so
$ ldd a.out
        linux-vdso.so.1 (0x00007ffede579000)
        libmystdio.so => /usr/local/lib/libmystdio.so (0x00007b98a6af5000)
        libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007b98a6800000)
        /lib64/ld-linux-x86-64.so.2 (0x00007b98a6b09000)
④ 创建软链接到系统库文件

适用于库文件需要经常更新,不想复制的场景。多个版本共存时,可以通过软链接切换。

# 1. 创建软链接(需要sudo权限)
$ sudo ln -s /home/zyt/linux-journey-log/code_25_5_15/dynamiclib/stdc/lib/libmystdio.so /usr/local/lib/libmystdio.so
[sudo] password for zyt: 
# 2. 更新动态链接器缓存
$ sudo ldconfig
# 3. 验证软链接
$ ls -l /usr/local/lib/libmystdio.so
lrwxrwxrwx 1 root root 74 May 15 12:06 /usr/local/lib/libmystdio.so -> /home/zyt/linux-journey-log/code_25_5_15/dynamiclib/stdc/lib/libmystdio.so
$ ldd a.out
        linux-vdso.so.1 (0x00007ffdfb1f9000)
        libmystdio.so => /usr/local/lib/libmystdio.so (0x0000701c1ab07000)
        libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x0000701c1a800000)
        /lib64/ld-linux-x86-64.so.2 (0x0000701c1ab1b000)

四、外部库(补充)

推荐一个好玩的图形库ncurses,使用指南:ncurse编程指南_ncurses教程-CSDN博客

// 安装
// Centos
$ sudo yum install -y ncurses-devel
// ubuntu
$ sudo apt install -y libncurses-dev

五、目标文件(原理部分)

编译和链接这两个步骤,在Windows下被我们的IDE封装的很完美,我们一般都是一键构建非常方便,但一旦遇到错误的时候呢,尤其是链接相关的错误,很多人就束手无策了。在Linux下,我们之前也学习过如何通过gcc编译器来完成这一系列操作。

接下来我们深入探讨一下编译和链接的整个过程,来更好地理解动静态库的使用原理。

先来回顾下什么是编译呢?编译的过程其实就是将我们程序的源代码翻译成CPU能够直接运行的机器代码。关键点:每个源文件独立编译,生成对应的目标文件。如果函数定义在其他文件中(如 hello.c 调用 code.c 中的 run()),编译器会暂时 标记未解析的符号(需链接阶段处理)。使用 -c 选项表示“只编译不链接”。

比如:在一个源文件hello.c里简单输出“hello world!”,并且调用一个run函数,而这个函数被定义在另一个原文件code.c中。这里我们就可以调用gcc -c来分别编译这两个原文件。

// hello.c
#include 
void run();
int main()
{
    printf("hello world!
");
    run();
    return 0;
}
// code.c
#include 
void run()
{
    printf("running...
");
}

$ gcc -c hello.c
$ gcc -c code.c
$ ls
code.c  code.o  hello.c  hello.o

可以看到,在编译之后会生成两个扩展名为.o的文件,它们被称为目标文件。要注意的是如果修改了一个原文件,那么只需要单独编译它这一个,而不需要浪费时间重新编译整个工程。目标文件是一个二进制的文件,文件的格式是ELF,是对二进制代码的一种封装。

# file命令用于辨识文件类型
$ file hello.o
hello.o: ELF 64-bit LSB relocatable, x86-64, version 1 (SYSV), not stripped
$ file code.o
code.o: ELF 64-bit LSB relocatable, x86-64, version 1 (SYSV), not stripped

六、EIL文件

1. ELF文件的四种格式

类型说明示例
可重定位文件包含代码和数据,需链接生成可执行文件或共享库.o 文件(hello.o
可执行文件可直接加载运行的程序,即可执行程序./a.out
共享目标文件动态链接库,运行时加载.so 文件(libc.so
核心转储文件进程崩溃时的内存快照(如 segmentation fault 生成),存放当前进程上下文,用于dump信号触发。core 文件

2. ELF文件的核心结构

ELF文件由以下四部分组成,可通过 readelf 工具查看。

(1)ELF Header(ELF头)

描述文件的主要特性。文件的类型(如可执行/共享库)、目标机器架构(如x86-64)、入口地址、节头表和程序头表的位置等。其位于文件的开始位置,它的主要目的是定位文件的其他部分。

$ readelf -h hello.o   # 查看目标文件的ELF头

关键字段:

Type: REL(可重定位文件)、EXEC(可执行文件)、DYN(共享库)。

Entry point address: 可执行文件的入口地址(如 _start)。

(2)Program Header Table(程序头表)

列举了所有有效的段(segments)和他们的属性。表里记着每个段的开始的位置和位移(offset)、长度,毕竟这些段,都是紧密的放在二进制文件中,需要段表的描述信息,才能把他们每个段分割开。作用:指导操作系统如何加载可执行文件或共享库(如哪些段需加载到内存、权限设置)。

注意:仅存在于可执行文件和共享库(可重定位文件如 .o 没有此表)。

$ readelf -l a.out     # 查看可执行文件的程序头表(segment)

关键段(Segments):

LOAD: 需加载到内存的代码段(.text)、数据段(.data、.bss)。

INTERP: 动态链接器路径(如 /lib64/ld-linux-x86-64.so.2)。

(3)Section Header Table(节头表)

描述所有节(Sections)的信息(位置、大小、类型),供链接和调试使用。

$ readelf -S hello.o   # 查看目标文件的节头表

关键节(Sections):

.text: 机器指令(代码),是程序的主要执行部分。

.data: 已初始化的全局/静态变量。

.bss: 未初始化的全局/静态变量(在文件中不占空间,预留位置,加载时清零)。

.symtab: 符号表(函数/变量名及其地址的对应关系)。

.rel.text: 重定位信息(需链接器修正的代码地址)。

(4)Sections(节)

ELF文件中的基本组成单位,包含了特定类型的数据。ELF文件的各种信息和数据都存储在不同的节中,如代码节存储了可执行代码,数据节存储了全局变量和静态数据等。

节与段的关系:

链接视角:使用节(如 .text、.data)。

执行视角:操作系统按段(如 LOAD)加载,一个段可能包含多个节(如将 .text 和 .rodata 合并到只读代码段)。稍后讲

七、ELF从加载到轮廓

1. ELF形成可执行文件

Step-1-编译:将 .c/.cpp 源代码文件编译成 可重定位目标文件。(预处理-编译-汇编)

Step-2-链接:将多个 .o 文件合并,生成 可执行文件(a.out) 或 共享库(.so)。

具体步骤

(1)符号解析(Symbol Resolution)

① 检查所有 .o 文件的 .symtab,确保每个符号(函数/变量)有且仅有一个定义。

② 如果某个符号未定义(如 printf),链接器会去 静态库(.a) 或 动态库(.so) 中查找。

(2)节(Section)合并

将多个 .o 文件的同名节合并:(也会涉及库的合并)

• 所有 .text → 合并到可执行文件的 .text

• 所有 .data → 合并到可执行文件的 .data

• 所有 .bss → 合并到可执行文件的 .bss

(3)重定位

① 修正代码和数据中的 地址引用(如 call printf 的真实地址)。

② 使用 .rel.text 和 .rel.data 表计算最终地址。

(4)生成可执行文件

最终生成 可执行 ELF 文件(a.out),包含:

• 程序头表(Program Header Table):告诉操作系统如何加载程序。

• 段(Segments):如 LOAD(代码段、数据段)、DYNAMIC(动态链接信息)。

2. ELF可执行文件加载

• 一个ELF会有多种不同的Section(节),在加载到内存的时候,也会进行Section合并,形成segment(段)。

• 合并原则:相同属性,比如:可读,可写,可执行,需要加载时申请空间等。【某些 Section(如 .debug_info)仅用于调试,不参与运行,因此不会被映射到任何 Segment。】

权限典型 Section合并后的 Segment
R E(可读可执行).text.plt.rodataLOAD 代码段
R W(可读可写).data.bss.gotLOAD 数据段
R(只读).eh_frame.dynstr可能合并到代码段

• 这样,即便是不同的Section,在加载到内存中,可能会以segment的形式,加载到一起。

• 很显然,这个合并工作也已经在形成ELF的时候,合并方式已经确定了(链接器(ld)根据链接脚本的规则合并 Section 为 Segment。),具体合并原则被记录在了ELF的程序头表(Program header table)中。

$ readelf -S hello.o   # 查看可执行程序的section
There are 14 section headers, starting at offset 0x298:

Section Headers:
  [Nr] Name              Type             Address           Offset
       Size              EntSize          Flags  Link  Info  Align
  [ 0]                   NULL             0000000000000000  00000000
       0000000000000000  0000000000000000           0     0     0
  [ 1] .text             PROGBITS         0000000000000000  00000040
       0000000000000028  0000000000000000  AX       0     0     1
  [ 2] .rela.text        RELA             0000000000000000  000001c0
       0000000000000048  0000000000000018   I      11     1     8
  [ 3] .data             PROGBITS         0000000000000000  00000068
       0000000000000000  0000000000000000  WA       0     0     1
  [ 4] .bss              NOBITS           0000000000000000  00000068
       0000000000000000  0000000000000000  WA       0     0     1
  [ 5] .rodata           PROGBITS         0000000000000000  00000068
       000000000000000d  0000000000000000   A       0     0     1
  [ 6] .comment          PROGBITS         0000000000000000  00000075
       000000000000002c  0000000000000001  MS       0     0     1
  [ 7] .note.GNU-stack   PROGBITS         0000000000000000  000000a1
       0000000000000000  0000000000000000           0     0     1
  [ 8] .note.gnu.pr[...] NOTE             0000000000000000  000000a8
       0000000000000020  0000000000000000   A       0     0     8
  [ 9] .eh_frame         PROGBITS         0000000000000000  000000c8
       0000000000000038  0000000000000000   A       0     0     8
  [10] .rela.eh_frame    RELA             0000000000000000  00000208
       0000000000000018  0000000000000018   I      11     9     8
  [11] .symtab           SYMTAB           0000000000000000  00000100
       00000000000000a8  0000000000000018          12     4     8
  [12] .strtab           STRTAB           0000000000000000  000001a8
       0000000000000017  0000000000000000           0     0     1
  [13] .shstrtab         STRTAB           0000000000000000  00000220
       0000000000000074  0000000000000000           0     0     1
Key to Flags:
  W (write), A (alloc), X (execute), M (merge), S (strings), I (info),
  L (link order), O (extra OS processing required), G (group), T (TLS),
  C (compressed), x (unknown), o (OS specific), E (exclude),
  D (mbind), l (large), p (processor specific)

$ readelf -l a.out     # 查看section合并后的segment

Elf file type is DYN (Position-Independent Executable file)
Entry point 0x1060
There are 13 program headers, starting at offset 64

Program Headers:
  Type           Offset             VirtAddr           PhysAddr
                 FileSiz            MemSiz              Flags  Align
  PHDR           0x0000000000000040 0x0000000000000040 0x0000000000000040
                 0x00000000000002d8 0x00000000000002d8  R      0x8
  INTERP         0x0000000000000318 0x0000000000000318 0x0000000000000318
                 0x000000000000001c 0x000000000000001c  R      0x1
      [Requesting program interpreter: /lib64/ld-linux-x86-64.so.2]
  LOAD           0x0000000000000000 0x0000000000000000 0x0000000000000000
                 0x0000000000000628 0x0000000000000628  R      0x1000
  LOAD           0x0000000000001000 0x0000000000001000 0x0000000000001000
                 0x0000000000000199 0x0000000000000199  R E    0x1000
  LOAD           0x0000000000002000 0x0000000000002000 0x0000000000002000
                 0x0000000000000124 0x0000000000000124  R      0x1000
  LOAD           0x0000000000002db8 0x0000000000003db8 0x0000000000003db8
                 0x0000000000000258 0x0000000000000260  RW     0x1000
  DYNAMIC        0x0000000000002dc8 0x0000000000003dc8 0x0000000000003dc8
                 0x00000000000001f0 0x00000000000001f0  RW     0x8
  NOTE           0x0000000000000338 0x0000000000000338 0x0000000000000338
                 0x0000000000000030 0x0000000000000030  R      0x8
  NOTE           0x0000000000000368 0x0000000000000368 0x0000000000000368
                 0x0000000000000044 0x0000000000000044  R      0x4
  GNU_PROPERTY   0x0000000000000338 0x0000000000000338 0x0000000000000338
                 0x0000000000000030 0x0000000000000030  R      0x8
  GNU_EH_FRAME   0x000000000000201c 0x000000000000201c 0x000000000000201c
                 0x000000000000003c 0x000000000000003c  R      0x4
  GNU_STACK      0x0000000000000000 0x0000000000000000 0x0000000000000000
                 0x0000000000000000 0x0000000000000000  RW     0x10
  GNU_RELRO      0x0000000000002db8 0x0000000000003db8 0x0000000000003db8
                 0x0000000000000248 0x0000000000000248  R      0x1

 Section to Segment mapping:
  Segment Sections...
   00     
   01     .interp 
   02     .interp .note.gnu.property .note.gnu.build-id .note.ABI-tag .gnu.hash .dynsym .dynstr .gnu.version .gnu.version_r .rela.dyn .rela.plt 
   03     .init .plt .plt.got .plt.sec .text .fini 
   04     .rodata .eh_frame_hdr .eh_frame 
   05     .init_array .fini_array .dynamic .got .data .bss 
   06     .dynamic 
   07     .note.gnu.property 
   08     .note.gnu.build-id .note.ABI-tag 
   09     .note.gnu.property 
   10     .eh_frame_hdr 
   11     
   12     .init_array .fini_array .dynamic .got 

为什么要将section合并成segment?

1. 减少内存碎片,提高页面对齐效率
内存按页(Page)管理:现代操作系统以固定大小的页(如 4KB)为单位管理内存。如果多个小 Section 分散加载,会导致内存浪费。

例子:

        .text(代码段)占用 4097 字节 → 需要 2 页(4096 + 1)。

        .rodata(只读数据)占用 512 字节 → 需要 1 页。

        未合并时:共占用 3 页(实际使用 4097 + 512 = 4609 字节,利用率仅 37.5%)。

        合并后:.text + .rodata 总大小 4609 字节 → 仅需 2 页(利用率提升至 56.3%)。

合并策略:将权限相同(如只读、可执行)的 Section 合并到一个 Segment,使它们在内存中连续存储,减少碎片。

2.  统一内存权限,简化操作系统加载
Section 的权限可能相同:

例如 :

        .text(代码)、.rodata(只读数据)都是 R-X(可读、可执行)。

        .data(全局数据)、.bss(未初始化数据)都是 RW-(可读、可写)。

        操作系统按 Segment 设置权限:如果每个 Section 单独映射,操作系统需要为每个小段设置权限(频繁的系统调用,效率低)。

合并后:只需为整个 Segment 设置一次权限(例如一个 LOAD Segment 包含所有 R-X 的 Section)。
 

3. 提升程序加载速度
减少内存映射次数:

        合并前:操作系统需为每个 Section 单独调用 mmap(或类似机制)。

        合并后:只需为少数几个 Segment 调用 mmap,减少系统开销。

降低页表项(PTE)压力:

每个内存映射需要占用页表条目,合并后减少条目数量,节省内核资源。
 

4. 动态链接的优化
动态库(如 libc.so)的依赖项:动态链接器(如 ld-linux.so)需要快速定位程序中的 .dynamic、.got.plt 等关键 Section。

通过将这些 Section 合并到明确的 Segment(如 DYNAMIC),链接器能直接遍历 Program Header Table,而无需解析所有 Section。

对于程序头表和节头表又有什么用呢,其实ELF文件提供了2个不同的视图/视角来让我们理解这两个部分:

  • 链接视图(Linking view) - 对应节头表 Section header table

    • 文件结构的粒度更细,将文件按功能模块的差异进行划分。静态链接分析的时候一般关注的是链接视图,能够理解ELF文件中包含的各个部分的信息。

    • 为了空间布局上的效率,将来在链接目标文件时,链接器会把很多节(section)合并,规整成可执行的段(segment)、可读写的段、只读段等。合并了后,空间利用率就高了。否则,很小的一段,未来物理内存页浪费太大(物理内存页分配一般都是整数倍一块给你,比如4k)。所以,链接器趁着链接就把小块们都合并了。

  • 执行视图(Execution view) - 对应程序头表 Program header table

    • 告诉操作系统,如何加载可执行文件,完成进程内存的初始化。一个可执行程序的格式中,一定有program header table。

说白了就是:一个在链接时作用,一个在运行加载时作用。

我们可以在ELF头中找到文件的基本信息,以及可以看到ELF头是如何定位程序头表和节头表的。例如我们查看下hello.o这个可重定位文件的主要信息: 

# 查看目标文件
$ readelf -h hello.o
ELF Header:
  Magic:    7f 45 4c 46 02 01 01 00 00 00 00 00 00 00 00 00  # ELF文件魔数标识
  Class:                             ELF64                    # 文件类型(64位)
  Data:                              2's complement, little endian  # 数据编码方式(小端序)
  Version:                           1 (current)              # ELF版本
  OS/ABI:                            UNIX - System V          # 操作系统ABI类型
  ABI Version:                       0                       # ABI版本
  Type:                              REL (Relocatable file)   # 文件类型(可重定位文件)
  Machine:                           Advanced Micro Devices X86-64  # 机器架构(x86-64)
  Version:                           0x1                     # 版本
  Entry point address:               0x0                     # 入口地址(目标文件为0)
  Start of program headers:          0 (bytes into file)      # 程序头表起始位置(目标文件无)
  Start of section headers:          728 (bytes into file)    # 节头表起始位置
  Flags:                             0x0                     # 处理器特定标志
  Size of this header:               64 (bytes)              # ELF头大小
  Size of program headers:           0 (bytes)               # 程序头表条目大小(目标文件无)
  Number of program headers:         0                       # 程序头表条目数(目标文件无)
  Size of section headers:           64 (bytes)              # 节头表条目大小
  Number of section headers:         13                      # 节头表条目数
  Section header string table index: 12                      # 节名称字符串表索引

# 查看可执行程序
$ gcc *.o
$ readelf -h a.out
ELF Header:
  Magic:    7f 45 4c 46 02 01 01 00 00 00 00 00 00 00 00 00  # ELF文件魔数标识
  Class:                             ELF64                    # 文件类型(64位)
  Data:                              2's complement, little endian  # 数据编码方式(小端序)
  Version:                           1 (current)              # ELF版本
  OS/ABI:                            UNIX - System V          # 操作系统ABI类型
  ABI Version:                       0                       # ABI版本
  Type:                              DYN (Shared object file) # 文件类型(动态共享对象)
  Machine:                           Advanced Micro Devices X86-64  # 机器架构(x86-64)
  Version:                           0x1                     # 版本
  Entry point address:               0x1060                  # 程序入口地址(_start)
  Start of program headers:          64 (bytes into file)     # 程序头表起始位置
  Start of section headers:          14768 (bytes into file)  # 节头表起始位置
  Flags:                             0x0                     # 处理器特定标志
  Size of this header:               64 (bytes)              # ELF头大小
  Size of program headers:           56 (bytes)              # 程序头表条目大小
  Number of program headers:         13                      # 程序头表条目数
  Size of section headers:           64 (bytes)              # 节头表条目大小
  Number of section headers:         31                      # 节头表条目数
  Section header string table index: 30                      # 节名称字符串表索引

八、理解链接与加载

1. 静态链接

• 无论是自己的.o还是静态库中的.o,本质都是把.o文件爱你进行链接的过程。

• 研究静态链接本质就是研究.o是如何链接的。

$ ls 
code.c  hello.c
$ gcc -c *.c
$ ls
code.c  code.o  hello.c  hello.o
$ gcc *.o -o main.exe
$ ll
total 40
drwxrwxr-x 2 zyt zyt  4096 May 17 15:57 ./
drwxrwxr-x 6 zyt zyt  4096 May 16 15:54 ../
-rw-rw-r-- 1 zyt zyt    62 May 16 15:55 code.c
-rw-rw-r-- 1 zyt zyt  1496 May 17 15:56 code.o
-rw-rw-r-- 1 zyt zyt   100 May 16 15:55 hello.c
-rw-rw-r-- 1 zyt zyt  1560 May 17 15:56 hello.o
-rwxrwxr-x 1 zyt zyt 16016 May 17 15:57 main.exe*

• objdump -d 命令:查看代码段(.text)的反汇编 

我们发现call指令转跳的地址都被设置成了0,这是为什么?

其实在编译hello.c的时候,编译器是不知道printf和run函数的存在的,因此,编译器只能将这两个函数的转跳地址先暂时设为0。直到链接的时候,为了让连接器将来在链接时,能正确定位到这些被修正的地址,在代码块(.data)中还存在一个重定位表,将来链接时,就会根据表里记录的地址将其修正。细节如下:

(1)编译阶段

当编译器处理 hello.c 时,如果遇到外部函数(如 printf 或 run),它的处理流程如下:

• 编译器仅知道这些符号的名称(如 printf),但不知道它们的实际地址或代码内容。因为这些符号可能定义在其他文件(如 libc.so 或 code.o)中。

• 生成占位符:对于函数调用(如 call printf),编译器会生成一个临时地址 00 00 00 00;对于数据引用(如 lea 0x0(%rip),%rax),同样填充零偏移。

• 保留重定位信息:编译器在目标文件(.o)中生成 重定位表(Relocation Table),记录哪些指令需要后续修正。

(2)重定位表

目标文件中包含一个或多个重定位表(如 .rela.text),用于指导链接器如何修正占位符。
通过 readelf -r hello.o 可以查看:

条目介绍:
Offset:占位符在 .text 节中的位置(如 0x0f 对应 call 的操作码位置)。

Type:重定位类型(如 R_X86_64_PC32 表示 32 位相对地址调用)。
Sym. Value :显示了符号的值。

Sym. Name:需要解析的符号名(run)。

Addend:修正时的附加偏移(通常为 -4,因为 RIP 相对寻址会指向下一条指令)。

上图显示证明:在hello.o的.rela.text节中,puts(就是printf)和run的Sym. Value都是0000000000000000,这表示这些符号在目标文件中尚未解析,即它们的地址被初始化为0。这表明这些符号在当前的目标文件中没有定义,需要在链接时从其他目标文件或库中解析,以确定它们的最终地址。

 (3)静态链接阶段:地址修正

链接器(如 ld)在合并所有目标文件时,会完成以下操作:

•  符号解析(Symbol Resolution)

在全局符号表中查找 printf 和 run 的定义:printf 通常来自 libc.a(静态库)或 libc.so(动态库)。run 来自 code.o。若符号未定义,链接器报错(undefined reference)。

• 分配最终地址

合并所有 .text 节,并为函数分配运行时地址。

• 修正占位符

根据重定位表,修改代码中的零地址为实际地址。

查看最终程序的反汇编,就能显示函数运行时的地址了:

 readelf -s main.exe 查看符号表
两个.o合并之后,在最终的程序中的符号表中,就找到了run;【0000000000001149】其实是地址,FUNC表示run符号类型是函数;【16】就是run函数所在的section被合并在最终的那一个section中了,16就是下标。

readelf -S main.exe:查看节区头表(Section Headers)
作用:显示文件的所有节区(Section)信息,描述各节区的布局和属性。

我们看到 code.o 和 hello.o 的 .text 合并后得到的是 main.exe 的第16个section。

(4)总结

所以链接其实就是将编译之后的所有目标文件连同用到的一些静态库运行时库组合,拼装成一个独立的可执行文件。其中就包括我们之前提到的地址修正,当所有模块组合在一起之后,链接器会根据我们的.o文件或者静态库中的重定位表找到那些需要被重定位的函数全局变量,从而修正它们的地址。这就是静态链接的过程。

链接过程中涉及到对.o中的外部符号进行地址重定位。

2. ELF加载和进程地址空间

(1)逻辑地址 / 虚拟地址 

• 一个ELF程序,在没有被加载到内存的时候,有没有地址呢?

有逻辑地址(虚拟地址布局),但无物理地址。当代计算机都采用“平坦模式”:现代计算机采用虚拟内存机制,编译器在生成 ELF 文件时,会按虚拟地址空间对代码和数据预编址(如 .text 从 0x400000 开始)。下面是objdump -d main.exe 反汇编后的代码:

通过 objdump -d 或 readelf -S 看到的地址是逻辑地址(起始地址+偏移量),表示该段代码/数据在进程虚拟空间中的预期位置。我们通常认为起始地址是0。也就是说,其实虚拟地址在程序还没有加载到内存的时候,就已经对可执行程序进行了统一编址。这些地址在链接阶段由链接器分配,基于链接脚本(Linker Script)的规则。

• 进程mm_struct、vm_area_struct在进程刚刚创建的时候,初始化数据从哪里来的?

数据来源:ELF 文件的 Program Header Table(即 Segment 信息)。程序头表描述了ELF文件中各个段的属性,包括它们的类型、文件偏移、虚拟地址、物理地址、大小等信息。操作系统利用这些信息来初始化进程的内存布局,包括设置页表、分配内存区域等。这些结构确保了进程的虚拟地址空间能够正确映射到物理内存,从而允许进程执行。

• 磁盘上的可执行程序,代码和数据编址其实就是虚拟地址的统一编址!

① 磁盘上的ELF地址是虚拟地址:由链接器按进程虚拟地址空间统一分配,保存在文件中。

② OS加载时按此布局映射:将ELF中的虚拟地址映射到进程的虚拟内存,再通过页表转为物理地址。

③ 协作机制:

        编译器:生成逻辑地址(目标文件)。

        链接器:统一分配虚拟地址(ELF文件)。

        操作系统:将虚拟地址映射到物理内存(运行时)。

(2)重新理解进程虚拟地址空间

ELF再被编译好之后,会把自己未来程序的入口地址记录在ELF header的Entry字段中:

作用:入口地址是操作系统加载程序后,CPU开始执行的第一条指令的虚拟内存地址。该地址通常指向程序初始化代码(如_start或.text段的起始位置),由链接器在静态链接阶段确定。
• 代码加载时,内核读取ELF头的Entry point address,将程序加载到内存后,跳转到入口地址执行。
• 页表映射时,内核将文件的.text段映射到物理内存,并建立页表项(虚拟→物理)。 

• 操作系统加载程序时的行为:
当用户运行程序时(如 ./a.out),操作系统(Linux 内核/Windows Loader)会:

解析可执行文件头部:读取 Entry point address,确定代码的起始虚拟地址。

分配虚拟地址空间:为进程创建页表,映射代码段(.text)、数据段(.data)等。

③ 设置程序计数器(PC/IP):将 CPU 的指令指针寄存器(x86: RIP,ARM: PC)指向入口地址。

(3)静态链接库在内存加载 

静态链接库在内存中的加载流程本质上是一个伪命题,因为其代码早已在编译期融入到了可执行文件之中。运行时,这类代码与其他自定义逻辑无异,均作为固定的部分参与程序的整体调度和执行。其具体特性如下:

① 静态链接库的特性

静态链接库的特点决定了它的加载行为不同于动态链接库。静态链接库在编译阶段即将库中的代码直接嵌入到目标文件中,这意味着最终生成的可执行文件本身已经包含了所有的库代码。

② 编译与链接阶段

• 在编译过程中,源代码被转换为目标文件(.o 或 .obj),其中包含汇编指令和符号表。

• 链接器负责解析未定义的符号,并将对应的实现从静态库中提取出来,将其实际代码复制到最终的可执行文件中。

此过程的关键在于,静态库中的代码并非以单独的形式存在于内存中,而是成为可执行文件的一部分。因此,静态链接库并没有传统意义上的“加载”概念,因为它已经在编译期间完成了集成。

③ 运行时的行为

当程序启动时,操作系统会为可执行文件分配一段连续的虚拟地址空间。这段地址空间包括以下几个区域:

• 代码区:存储程序的机器码,这部分内容来源于原始的源代码以及静态链接库中的代码片段。

• 数据区:分为初始化的数据段(如全局变量)和未初始化的数据段(BSS 段)。

• 堆栈区:用于动态分配内存和函数调用时的局部变量存储。

由于静态链接库的代码已经被完全嵌入到可执行文件中,因此在运行时不需要额外的操作系统介入来加载这些库代码。换句话说,静态链接库的代码已经是可执行文件的一部分,随同其他代码一起被映射到进程的地址空间。

④ 内存占用分析

尽管静态链接库避免了运行时依赖问题,但它也带来了显著的空间开销。每当一个新的应用程序使用相同的静态库时,该库的全部代码会被再次复制到新的可执行文件中。这种机制可能导致多个程序在内存中有重复的库副本,增加了整体系统的内存消耗。

3. 动态链接与动态库加载

(1)动态库加载

① 虚拟内存映射:

• 当进程A启动并需要加载动态库(如XXX.so)时,操作系统不会立即将整个库加载到物理内存

• 而是通过mm_struct(内存描述符)在进程的虚拟地址空间中建立映射关系

② 共享区(Shared Area):

• 动态库被映射到进程虚拟地址空间的"共享区"

• 这个区域与进程的"数据区"和"代码区"是分开的

• 多个进程可以共享同一个动态库的物理内存副本

③ 页表机制:

• 操作系统通过页表将虚拟地址映射到物理内存

• 对于动态库,这种映射是"按需"建立的 - 只有实际访问的部分才会被加载到物理内存

(2)进程间共享动态库

① 虚拟内存映射:

• 进程A和进程B各自有独立的虚拟地址空间

• 通过各自的mm_struct(内存描述符),两个进程都将XXX.so映射到自己的地址空间的"共享区"

• 虽然虚拟地址可能不同,但最终指向相同的物理内存区域

② 物理内存共享:

• 在物理内存中,XXX.so只有一份副本

• 两个进程的页表条目都指向这同一块物理内存区域

• 这是通过操作系统的内存管理实现的

(3)动态链接

① 动态链接概要

我们可以将需要共享的代码单独提取出来,保存成一个独立的动态链接库,等到程序运行的时候再将它们加载到内存,这样不但可以节省空间,因为同一个模块在内存中只需要保留一份副本,可以被不同的进程所共享。

动态链接实际上是将链接的整个过程推迟到了程序加载的时候。比如我们去运行一个程序,操作系统会首先将程序的数据代码连同它用到的一系列动态库先加载到内存,其中每个动态库的加载地址都是不固定的,操作系统会根据当前地址空间的使用情况为它们动态分配一段内存。

当动态库被加载到内存以后,一旦它的内存地址被确定,我们就可以去修正动态库中的那些函数跳转地址了。

② 可执行程序被编译器动了手脚

在C/C++程序中,当程序开始执行时,它首先并不会直接跳转到main函数。实际上,程序的入口点是_start,这是一个由C运行时库(通常是glibc)或链接器(如ld)提供的特殊函数。在_start函数中,会执行一系列初始化操作,这些操作包括:

• 设置堆栈:为程序创建一个初始的堆栈环境。

• 初始化数据段:将程序的数据段(如全局变量和静态变量)从初始化数据段复制到相应的内存位置,并清零未初始化的数据段。

• 动态链接:这是关键的一步,_start函数会调用动态链接器的代码来解析和加载程序所依赖的动态库(shared libraries)。动态链接器会处理所有的符号解析和重定位,确保程序中的函数调用和变量访问能够正确地映射到动态库中的实际地址。动态链接器(如ld-linux.so)负责在程序运行时加载动态库。

动态连接器

标红的 /lib64/ld-linux-x86-64.so.2 (0x00007f42c02b6000) 表示动态链接器(Dynamic Linker/Loader) 的路径和内存映射地址。

• 内核首先加载动态链接器到内存(而非直接加载程序)。

• 当程序启动时,动态链接器会解析程序中的动态库依赖,并加载这些库到内存中。

• Linux系统通过环境变量(如LD_LIBRARY_PATH)和配置文件(如/etc/ld.so.conf及其子配置文件)来指定动态库的搜索路径。

• 这些路径会被动态链接器在加载动态库时搜索。

• 为了提高动态库的加载效率,Linux系统会维护一个名为/etc/ld.so.cache的缓存文件。

• 该文件包含了系统中所有已知动态库的路径和相关信息,动态链接器在加载动态库时会首先搜索这个缓存文件。

• 库搜索顺序:

• 调用__libc_start_main:一旦动态链接完成,_start函数会调用__libc_start_main(这是glibc提供的一个函数)。__libc_start_main函数负责执行一些额外的初始化工作,比如设置信号处理函数、初始化线程库(如果使用了线程)等。

• 调用main函数:最后,__libc_start_main函数会调用程序的main函数,此时程序的执行控制权才正式交给用户编写的代码。

• 处理main函数返回值:当main函数返回时,__libc_start_main会负责处理这个返回值,并最终调用_exit函数来终止程序。

③ 动态库中的相对地址

• 动态库是采用相对编址(位置无关代码,PIC)的方案进行编址的,这种机制使得动态库可以被加载到进程地址空间的任意位置而无需重写代码。

• 位置无关代码(PIC):是通过所有地址引用都使用相对偏移量,而非绝对地址实现的,使代码无论加载到内存哪个位置都能正确执行(而静态链接的程序使用固定绝对地址,加载位置固定)。

• 平坦内存模式:采用统一编址,使整个地址空间是一个连续的线性空间。而在动态库视角,每个库都"认为"自己从地址0开始,实际通过偏移量计算真实地址。可执行程序(.exe)和动态库都要遵守“平坦模式”,只不过.exe是直接加载的,动态库需要动态加载。

④ 程序与动态库映射

• 动态库也是一个文件,要访问也是要被先加载,要加载也是要被打开的。

• 让我们的进程找到动态库的本质:也是文件操作,不过我们访问库函数,通过虚拟地址进行跳转访问的,所以需要把动态库映射到进程的地址空间中

⑤ 程序调用库函数

已知库的虚拟起始地址和函数偏移量的情况下,访问库中所有方法都需要:函数绝对地址 = 库虚拟起始地址 + 函数偏移量。并且,整个调用过程是从代码段转跳到共享区,调用完毕后返回代码区,整个过程都在进程地址空间中进行。

⑥ 全局偏移量表GOT

注意:也就是说,我们的程序运行之前,先把所有库加载并映射,所有库的起始虚拟地址都应该提前知道,然后对我们加载到内存中的程序的库函数调用进行地址修改,在内存中二次完成地址设置(这个叫做加载地址重定位) , 等等,不是说代码区在进程中是只读的吗?怎么修改?能修改吗?

所以:动态链接采用的做法是在 .data(.data是可读写的,支持动态修改)中专门预留一片区域用来存放函数的跳转地址,它也被叫做全局偏移表GOT,表中每一项都是本运行模块要引用的一个全局变量或函数的地址。代码段通过相对寻址访问GOT表项,动态链接器在加载时填充GOT表中的实际地址。

• GOT运行工作流程:

首次调用:
* 调用PLT跳转到动态链接器
* 解析符号得到真实地址并填充GOT表
* 跳转到真实函数地址
后续调用:
* 直接通过GOT表跳转(无解析开销)

•  GOT表不共享:由于GOT表项必须包含当前进程的绝对地址,并且不同进程的地址空间中,各动态库的绝对地址、相对位置都不同。

• 在单个.so下,由于GOT表与 .text 的相对位置是固定的,我们完全可以利用CPU的相对寻址来找到GOT表。

• 这种方式实现的动态链接就被叫做 PIC 地址无关代码 。换句话说,我们的动态库不需要做任何修改,被加载到任意内存地址都能够正常运行,并且能够被所有进程共享,这也是为什么之前我们给编译器指定-fPIC参数的原因,PIC=相对编址+GOT。

备注:PLT是什么?

PLT(过程链接表)是动态链接的核心机制之一,主要用于延迟绑定,即在程序运行时按需解析动态库函数的真实地址,而不是在程序启动时就解析所有函数。
解决动态库函数调用问题:动态库的函数地址在编译时是未知的(因为库可能被加载到任意地址),PLT 提供了一种间接跳转机制,使得程序可以先调用 PLT 条目,再由 PLT 负责跳转到正确的函数地址。优化后,采用延迟绑定,只有在函数第一次被调用时才会解析其真实地址,后续调用直接跳转,避免启动时解析所有符号的开销。

⑦ 动态库间依赖

• 库间依赖通过动态链接器递归加载:动态库(.so)也可以依赖其他库,库间依赖通过动态链接器递归加载,保证所有库的 GOT 表被正确填充。

PIC + GOT/PLT 机制:动态链接器递归加载,加载 libA.so 时,发现它依赖 libB.so,动态链接器会先加载 libB.so 并修正 libA.so 的 GOT 表。如果 libB.so 又依赖 libC.so,则继续递归加载。

• ELF 格式统一:所有库都是ELF格式,结构一致(都有 .got、.plt、.dynamic),确保动态链接器能统一处理。库的代码段(.text)仍然是位置无关(PIC),通过 %rip 相对寻址访问自己的 GOT 表。每个库的 GOT 表是独立的,动态链接器会分别填充。

 解决依赖关系的时候,就是加载并完善互相之间的GOT表的过程。

总而言之,动态链接实际上将链接的整个过程,比如符号查询、地址的重定位从编译时推迟到了程序的运行时,它虽然牺牲了一定的性能和程序加载时间,但绝对是物有所值的。因为动态链接能够更有效地利用磁盘空间和内存资源,以极大方便了代码的更新和维护,更关键的是,它实现了二进制级别的代码复用。

4. 总结

• 静态链接的出现,提高了程序的模块化水平。对于一个大的项目,不同的人可以独立地测试和开发自己的模块。通过静态链接,生成最终的可执行文件。

• 我们知道静态链接会将编译产生的所有目标文件,和用到的各种库合并成一个独立的可执行文件,其中我们会去修正模块间函数的跳转地址,也被叫做编译重定位(也叫做静态重定位)。

• 而动态链接实际上将链接的整个过程推迟到了程序加载的时候。比如我们去运行一个程序,操作系统会首先将程序的数据代码连同它用到的一系列动态库先加载到内存,其中每个动态库的加载地址都是不固定的,但是无论加载到什么地方,都要映射到进程对应的地址空间,然后通过.GOT方式进行调用(运行重定位,也叫做动态地址重定位)。

本文地址:https://www.vps345.com/11431.html

搜索文章

Tags

PV计算 带宽计算 流量带宽 服务器带宽 上行带宽 上行速率 什么是上行带宽? CC攻击 攻击怎么办 流量攻击 DDOS攻击 服务器被攻击怎么办 源IP docker 容器 运维 java-rabbitmq java 服务器安全 网络安全策略 防御服务器攻击 安全威胁和解决方案 程序员博客保护 数据保护 安全最佳实践 人工智能 自动化 centos python linux 机器学习 服务器 深度学习 游戏 云计算 Deepseek Deepseek-R1 大模型 私有化部署 推理模型 deepseek DeepSeek-R1 API接口 c++ javascript 前端 chrome edge 网络工程师 网络管理 软考 2024 2024年上半年 下午真题 答案 数据库 oracle 关系型 安全 分布式 tcp/ip 网络 服务器无法访问 ip地址无法访问 无法访问宝塔面板 宝塔面板打不开 DNS 神经网络 计算机视觉 卷积神经网络 redis ddos android 网络安全 web安全 harmonyos 华为 https ubuntu ssh 物联网 arm开发 架构 无人机 机器人 springsecurity6 oauth2 授权服务器 前后端分离 笔记 C 环境变量 进程地址空间 进程 FTP服务器 数据库系统 RTSP xop RTP RTSPServer 推流 视频 nginx 负载均衡 vue.js audio vue音乐播放器 vue播放音频文件 Audio音频播放器自定义样式 播放暂停进度条音量调节快进快退 自定义audio覆盖默认样式 YOLO conda pytorch vscode yolov5 git npm php 开发语言 github flutter Google pay Apple pay Ollama 模型联网 API CherryStudio CH340 单片机 嵌入式硬件 串口驱动 CH341 uart 485 ai nlp Flask FastAPI Waitress Gunicorn uWSGI Uvicorn django AIGC 人工智能生成内容 Dify 网络结构图 YOLOv8 NPU Atlas800 A300I pro asi_bench oracle fusion oracle中间件 华为云 华为od 网络协议 windows spring boot macos outlook 错误代码2603 无网络连接 2603 游戏引擎 学习 c语言 agi Ubuntu DeepSeek DeepSeek Ubuntu DeepSeek 本地部署 DeepSeek 知识库 DeepSeek 私有化知识库 本地部署 DeepSeek DeepSeek 私有化部署 指令 kubernetes prometheus grafana 云原生 node.js 操作系统 进程控制 Ubuntu pycharm ide 策略模式 英语 学习方法 经验分享 程序人生 rpc 远程桌面 Qwen2.5-coder 离线部署 gnu mysql 镜像源 unix 语言模型 计算机外设 bug ROS 自动驾驶 udp gcc centos 7 ip协议 MacMini Mac 迷你主机 mini Apple llama deepseek-r1 大模型本地部署 mcp iventoy VmWare OpenEuler 算法 opencv 自然语言处理 apache CPU 使用率 系统监控工具 linux 命令 adb 交互 grub 版本升级 扩容 mount挂载磁盘 wrong fs type LVM挂载磁盘 Centos7.9 cron crontab日志 stm32 银河麒麟 智能路由器 dell服务器 服务器配置 生物信息学 本地环回 bind tomcat 腾讯云 mysql离线安装 ubuntu22.04 mysql8.0 快捷键 旋转屏幕 自动操作 科技 个人开发 react.js 前端面试题 持续部署 react next.js 部署 部署next.js USB转串口 数据分析 jvm 虚拟机 typescript 鸿蒙 qt arm 外网访问 内网穿透 端口映射 交换机 硬件 设备 GPU PCI-Express ACL 流量控制 基本ACL 规则配置 iftop 网络流量监控 dify 知识库 本地化部署 qps 高并发 opengl adobe AutoDL ArkUI 鸿蒙系统 ArkTS 信息与通信 fstab axure 富文本编辑器 ui ollama 压力测试 大模型压力测试 EvalScope 智能体开发 AI 环境迁移 缓存 ssh漏洞 ssh9.9p2 CVE-2025-23419 ip gitea UEFI Legacy MBR GPT U盘安装操作系统 Linux xrdp 远程连接 cursor kafka 远程工作 程序 编程 内存 性能分析 spring cloud compose Ubuntu20.04 GLIBC 2.35 面试 职场和发展 阿里云 ipython RAID RAID技术 磁盘 存储 selenium 性能优化 svn 数据挖掘 jellyfin nas VMware 虚拟现实 vite vue3 SRS 流媒体 直播 glibc c# 开源 milvus 数据结构 postman 测试工具 并查集 leetcode centos-root /dev/mapper yum clean all df -h / du -sh intellij-idea rust腐蚀 jenkins maven virtualenv excel 在线预览 xlsx xls文件 在浏览器直接打开解析xls表格 前端实现vue3打开excel 文件地址url或接口文档流二进 像素流送api 像素流送UE4 像素流送卡顿 像素流送并发支持 gitlab unity gitee 浪潮信息 AI服务器 filezilla 无法连接服务器 连接被服务器拒绝 vsftpd 331/530 pygame 实时音视频 音视频 kylin Doris搭建 docker搭建Doris Doris搭建过程 linux搭建Doris Doris搭建详细步骤 Doris部署 后端 阻塞队列 生产者消费者模型 服务器崩坏原因 http asp.net大文件上传 asp.net大文件上传下载 asp.net大文件上传源码 ASP.NET断点续传 华为OD 华为OD机试真题 可以组成网络的服务器 IMM spring rsyslog devops elasticsearch IM即时通讯 QQ 微信 企业微信 剪切板对通 HTML FORMAT ansible mcu linux环境变量 vnc MCP 串口服务器 eureka 雨云 NPS Linux 维护模式 目标检测 fastapi mcp-proxy mcp-inspector fastapi-mcp agent sse 灵办AI debian PVE 网络药理学 生信 gromacs 分子动力学模拟 MD 动力学模拟 JDK Java LInux Windows 自定义客户端 SAS visualstudio AI编程 ffmpeg 视频编解码 pip C语言 计算机网络 live555 rtsp rtp rdp 远程服务 microsoft word图片自动上传 word一键转存 复制word图片 复制word图文 复制word公式 粘贴word图文 粘贴word公式 教程 环境搭建 Maven k8s 大数据 spark hive rocketmq uni-app dash json 正则表达式 FTP 服务器 开发环境 burpsuite 安全工具 mac安全工具 burp安装教程 渗透工具 工具分享 matplotlib fonts-noto-cjk ubuntu 18.04 安装教程 Alexnet flask 媒体 实时互动 list websocket 嵌入式 rtsp服务器 rtsp server android rtsp服务 安卓rtsp服务器 移动端rtsp服务 大牛直播SDK 镜像下载 freebsd 虚拟显示器 远程控制 python2 ubuntu24.04 VMware安装mocOS macOS系统安装 grep 目标跟踪 OpenVINO 推理应用 课程设计 openvpn server openvpn配置教程 centos安装openvpn VMware Tools vmware tools安装 vmwaretools安装步骤 vmwaretools安装失败 vmware tool安装步骤 vm tools安装步骤 vm tools安装后不能拖 vmware tools安装步骤 Cursor IIS .net core Hosting Bundle .NET Framework vs2022 IIS服务器 IIS性能 日志监控 bash 区块链 行情服务器 股票交易 速度慢 切换 股票量化接口 股票API接口 zabbix mariadb ubuntu20.04 ros ros1 Noetic 20.04 apt 安装 华为认证 计算生物学 生物信息 基因组 Ubuntu 22.04 MySql 算家云 算力租赁 rust stm32项目 ue4 着色器 ue5 虚幻 私有化 本地部署 postgresql zotero WebDAV 同步失败 代理模式 反向代理 宝塔面板访问不了 宝塔面板网站访问不了 宝塔面板怎么配置网站能访问 宝塔面板配置ip访问 宝塔面板配置域名访问教程 宝塔面板配置教程 硬件工程 大语言模型 LLMs webdav windows 服务器安装 网易邮箱大师 运维开发 eclipse 开机黑屏 编辑器 vim EtherNet/IP串口网关 EIP转RS485 EIP转Modbus EtherNet/IP网关协议 EIP转RS485网关 EIP串口服务器 1024程序员节 微信小程序 miniapp 真机调试 调试 debug 断点 网络API请求调试方法 集成学习 集成测试 链表 防火墙 端口号 开放端口 访问列表 DeepSeek C++ WSL2 IP 地址 MobaXterm 文件传输 电脑 NVML nvidia-smi 统信 国产操作系统 虚拟机安装 建站 Chatbox jupyter LDAP samba jar 孤岛惊魂4 nohup 异步执行 AI-native gpt HarmonyOS Next DevEco Studio asm 爬虫 图像处理 sql gpu算力 mongodb Hyper-V WinRM TrustedHosts GPU环境配置 Ubuntu22 CUDA PyTorch Anaconda安装 高级IO epoll VM搭建win2012 win2012应急响应靶机搭建 攻击者获取服务器权限 上传wakaung病毒 应急响应并溯源 挖矿病毒处置 应急响应综合性靶场 firewall efficientVIT YOLOv8替换主干网络 TOLOv8 上传视频文件到服务器 uniApp本地上传视频并预览 uniapp移动端h5网页 uniapp微信小程序上传视频 uniapp app端视频上传 uniapp uview组件库 mac ollama api ollama外网访问 wsl ssrf 失效的访问控制 hadoop chatgpt oneapi llm transformer 命名管道 客户端与服务端通信 系统安全 进程程序替换 execl函数 execv函数 execvp函数 execvpe函数 putenv函数 直播推流 Qualcomm WoS Python QNN AppBuilder pillow 驱动开发 升级 CVE-2024-7347 漏洞 小程序 微信小程序域名配置 微信小程序服务器域名 微信小程序合法域名 小程序配置业务域名 微信小程序需要域名吗 微信小程序添加域名 多层架构 解耦 linux驱动开发 C# MQTTS 双向认证 emqx 前端框架 perf TCP WebServer 微信分享 Image wxopensdk QT 5.12.12 QT开发环境 Ubuntu18.04 cuda golang 虚拟局域网 EtherCAT转Modbus EtherCAT转485网关 ECT转485串口服务器 ECT转Modbus485协议 ECT转Modbus串口网关 ECT转Modbus串口服务器 一切皆文件 iot AI大模型 大模型技术 本地部署大模型 maxkb ARG 知识图谱 LLM 大模型教程 ssl cpu 实时 使用 冯诺依曼体系 多线程服务器 Linux网络编程 aws nvm remote-ssh windwos防火墙 defender防火墙 win防火墙白名单 防火墙白名单效果 防火墙只允许指定应用上网 防火墙允许指定上网其它禁止 kali 共享文件夹 7z 硅基流动 ChatBox 桥接模式 vmware windows虚拟机 虚拟机联网 Kylin-Server 服务器安装 KingBase 大模型应用 Cline linuxdeployqt 打包部署程序 appimagetool kotlin android studio iphone 3d 数学建模 etcd 数据安全 RBAC openssl 网络爬虫 oceanbase rc.local 开机自启 systemd 麒麟 Trae IDE AI 原生集成开发环境 Trae AI express p2p visual studio code web 常用命令 文本命令 目录命令 mock mock server 模拟服务器 mock服务器 Postman内置变量 Postman随机数据 NFS JAVA yolov8 localhost termux Linux PID .netcore 系统 黑苹果 openEuler 安卓模拟器 智能合约 哈希算法 高效I/O 磁盘挂载 新盘添加 partedUtil clickhouse 读写锁 Invalid Host allowedHosts vue 毕设 命令 python3.11 pyside6 界面 webrtc 云电竞 云电脑 todesk 安全威胁分析 MCP server C/S diskgenius CNNs 图像分类 移动端开发 国产数据库 瀚高数据库 数据迁移 下载安装 ragflow html ESP32 lighttpd安装 Ubuntu配置 Windows安装 性能测试 服务器优化 Hive环境搭建 hive3环境 Hive远程模式 googlecloud rustdesk Redis Desktop vSphere vCenter 软件定义数据中心 sddc DevOps 软件交付 数据驱动 应用场景 思科 CPU 主板 电源 网卡 昇腾 大模型训练/推理 推理问题 mindie ocr 硬件架构 系统架构 jmeter 软件测试 macbook Docker Desktop cnn gateway Clion Nova ResharperC++引擎 Centos7 远程开发 java-ee shell脚本免交互 expect linux免交互 王者荣耀 html5 firefox springboot rag 大模型部署 kubeless zip unzip java-rocketmq 微服务 Kali Linux 黑客 渗透测试 信息收集 tcpdump 实验 n8n 工作流 ruoyi xcode intellij idea numpy 大文件分片上传断点续传及进度条 如何批量上传超大文件并显示进度 axios大文件切片上传详细教 node服务器合并切片 vue3大文件上传报错提示错误 vu大文件秒传跨域报错cors vasp安装 vr arkUI arkTs web3.py jdk tcp pdf open webui MacOS录屏软件 ip命令 新增网卡 新增IP 启动网卡 源码剖析 rtsp实现步骤 流媒体开发 多线程 智能手机 高效日志打印 串口通信日志 服务器日志 系统状态监控日志 异常记录日志 Ubuntu22.04 虚拟化 开发人员主页 Maxkb RAG技术 本地知识库 flash-attention 报错 ci/cd 代码规范 big data 大模型入门 burp suite 抓包 IPMI WLAN iTerm2 notepad SSH Xterminal Mermaid 可视化图表 自动化生成 ecmascript nextjs reactjs sqlserver 程序员 ufw 压测 ECS camera Arduino 电子信息 Agent CrewAI 大模型微调 GCC aarch64 编译安装 HPC 信号处理 MQTT 消息队列 安卓 智慧农业 开源鸿蒙 团队开发 监控k8s集群 集群内prometheus sublime text PX4 ios shell 话题通信 服务通信 聚类 卡死 软件需求 go file server http server web server frp 回显服务器 UDP的API使用 chatbox ai小智 语音助手 ai小智配网 ai小智教程 智能硬件 esp32语音助手 diy语音助手 matlab powerpoint 通信工程 毕业 kernel RustDesk自建服务器 rustdesk服务器 docker rustdesk css css3 GaN HEMT 氮化镓 单粒子烧毁 辐射损伤 辐照效应 HTML audio 控件组件 vue3 audio音乐播放器 Audio标签自定义样式默认 vue3播放音频文件音效音乐 自定义audio播放器样式 播放暂停调整声音大小下载文件 录音麦克风权限判断检测 录音功能 录音文件mp3播放 小程序实现录音及播放功能 RecorderManager 解决录音报错播放没声音问题 KylinV10 麒麟操作系统 Vmware 执法记录仪 智能安全帽 smarteye Linux Vim 实战案例 ECT转Modbus协议 EtherCAT转485协议 ECT转Modbus网关 HP Anyware 进程优先级 调度队列 进程切换 es6 qt6.3 g726 其他 yum docker-compose docker compose odoo 服务器动作 Server action 管道 匿名管道 openjdk dubbo protobuf 序列化和反序列化 安装 传统数据库升级 银行 磁盘监控 券商 股票交易接口api 类型 特点 温湿度数据上传到服务器 Arduino HTTP 蓝桥杯C++组 蓝桥杯 嵌入式实习 程序员创富 cmos WSL resolv.conf 深度求索 私域 工具 进程间通信 SSH 密钥生成 SSH 公钥 私钥 生成 vmamba seatunnel 腾讯云大模型知识引擎 Linux无人智慧超市 LInux多线程服务器 QT项目 LInux项目 单片机项目 网卡的名称修改 eth0 ens33 nohup后台启动 keepalived 弹性计算 云服务器 裸金属服务器 弹性裸金属服务器 源代码管理 web3 区块链项目 云原生开发 K8S k8s管理系统 NAT转发 NAT Server 显示器 NAS Termux Samba su sudo sudo原理 su切换 语音识别 计算机科学与技术 docker搭建nacos详解 docker部署nacos docker安装nacos 腾讯云搭建nacos centos7搭建nacos 产品经理 Reactor 设计模式 中间件 迁移指南 autoware ros2 开发 langchain deep learning nuxt3 飞腾处理器 国产化 .net 多产物 YOLOv12 arcgis 宝塔 DrissionPage linux内核 trae 服务器扩容没有扩容成功 软件工程 教育电商 游戏程序 wps elk iperf3 带宽测试 HiCar CarLife+ CarPlay QT RK3588 r语言 数据可视化 ROS2 lvm 磁盘分区 电路仿真 multisim 硬件工程师 硬件工程师学习 电路图 电路分析 仪器仪表 umeditor粘贴word ueditor粘贴word ueditor复制word ueditor上传word图片 ueditor导入word ueditor导入pdf ueditor导入ppt 客户端 alias unalias 别名 Claude Desktop Claude MCP Windows Cli MCP 安装MySQL 论文阅读 显卡驱动持久化 GPU持久化 桌面环境 混合开发 环境安装 僵尸进程 Kali 渗透 服务器部署 本地拉取打包 VMware安装Ubuntu Ubuntu安装k8s 可信计算技术 源码 毕业设计 海康 智能电视 webgl nacos ragflow 源码启动 豆瓣 追剧助手 迅雷 密码学 信息可视化 echarts deepseak 文心一言 豆包 KIMI 腾讯元宝 unity3d pyautogui cudnn anaconda 互信 bot Docker OpenCore H3C Trae叒更新了? dns linux上传下载 弹性服务器 京东云 RDP 云服务 Docker Compose Kubernetes HCIE 数通 Web应用服务器 宝塔面板无法访问 Mac软件 IP配置 netplan harmonyOS面试题 热榜 webpack 大屏端 个人博客 进程信号 联网 easyconnect 代理 软件构建 kamailio sip VoIP Linux24.04 deepin 设置代理 实用教程 沙盒 需求分析 提示词 电脑桌面出现linux图标 电脑桌面linux图标删除不了 电脑桌面Liunx图标删不掉 linux图标删不掉 离线部署dify 图文教程 VMware虚拟机 macOS系统安装教程 macOS最新版 虚拟机安装macOS Sequoia Carla 智能驾驶 nvidia TrueLicense ukui 麒麟kylinos openeuler Netty 安全漏洞 信息安全 WebUI DeepSeek V3 ESXi GameFramework HybridCLR Unity编辑器扩展 自动化工具 VPS gpt-3 权限 kylin v10 麒麟 v10 mac设置host k8s部署 MySQL8.0 高可用集群(1主2从) edge浏览器 onlyoffice 镜像 用户管理 视频监控 网站 WebRTC log4j 远程过程调用 Windows环境 sysctl.conf vm.nr_hugepages 单一职责原则 网络编程 聊天服务器 套接字 Socket 考试 threejs 3D Attention NLP iis DocFlow stable diffusion mamba Vmamba VMware创建虚拟机 k8s集群资源管理 sqlite3 物联网开发 Linux的权限 服务器数据恢复 数据恢复 存储数据恢复 raid5数据恢复 磁盘阵列数据恢复 redhat 银河麒麟操作系统 EMQX 通信协议 GPU训练 c ShapeFile GeoJSON Nginx Docker Hub docker pull daemon.json kind 软件卸载 系统清理 bcompare Beyond Compare laravel 高德地图 鸿蒙接入高德地图 HarmonyOS5.0 监控k8s 监控kubernetes AnythingLLM AnythingLLM安装 内网环境 visual studio api 搜索引擎 全文检索 图搜索算法 React Next.js 开源框架 DigitalOcean GPU服务器购买 GPU服务器哪里有 GPU服务器 数据集 rabbitmq WebVM MAVROS 四旋翼无人机 基础环境 自动化运维 流水线 脚本式流水线 tar 大文件秒传跨域报错cors AD域 photoshop 键盘 智能体 autogen openai coze zerotier openwrt 中兴光猫 换光猫 网络桥接 自己换光猫 嵌入式系统开发 软件开发 信任链 windows日志 fpga开发 kvm 锁屏不生效 创意 社区 mysql安装报错 windows拒绝安装 MS Materials db database docker run 数据卷挂载 交互模式 Windsurf conda配置 conda镜像源 ftp服务 文件上传 自动化编程 make命令 makefile文件 换源 国内源 Debian 日志分析 系统取证 pyicu SSE armbian u-boot 隐藏文件 scikit-learn 图论 netty moveit 机器人运动 rancher 状态管理的 UDP 服务器 Arduino RTOS win11 无法解析服务器的名称或地址 zookeeper selete ftp 具身智能 Isaac Sim 虚拟仿真 gradle WSL2 上安装 Ubuntu SecureCRT 分析解读 主从复制 Docker引擎已经停止 Docker无法使用 WSL进度一直是0 镜像加速地址 sublime text3 export import save load 迁移镜像 统信UOS bonding 链路聚合 云计算面试题 低代码 单例模式 网络用户购物行为分析可视化平台 大数据毕业设计 UOS 统信操作系统 Office xml LLM Web APP Streamlit initramfs Linux内核 Grub Mac内存不够用怎么办 数码 联机 僵尸毁灭工程 游戏联机 开服 openstack Xen KVM llama3 Chatglm 开源大模型 iBMC UltraISO EVE-NG nftables Linux权限 权限命令 特殊权限 拓扑图 5G vm 笔灵AI AI工具 Dell R750XS TRAE 域名服务 DHCP 符号链接 配置 P2P HDLC VSCode 测试用例 功能测试 动静态库 内网渗透 靶机渗透 v10 软件 矩阵 env 变量 ICMP 框架搭建 ECS服务器 fork wait waitpid exit VPN wireguard GRE 政务 分布式系统 监控运维 Prometheus Grafana 推荐算法 minio 显卡驱动 AP配网 AK配网 小程序AP配网和AK配网教程 WIFI设备配网小程序UDP开 MVS 海康威视相机 输入法 空间 查错 博客 IPv4 子网掩码 公网IP 私有IP 远程 执行 sshpass 操作 负载测试 显示管理器 lightdm gdm xshell termius iterm2 Echarts图表 折线图 柱状图 异步动态数据 鸿蒙开发 可视化效果 element-ui 上传视频并预览视频 vue上传本地视频及进度条功能 vue2选择视频上传到服务器 upload上传视频组件插件 批量上传视频 限制单个上传视频 金融 neo4j 数据仓库 数据库开发 数据库架构 重启 排查 系统重启 日志 原因 jetty undertow springboot容器部署 springboot容器化部署 微服务容器化负载均衡配置 微服务容器多节点部署 微服务多节点部署配置负载均衡 实时内核 序列化反序列化 双系统 ArcTS 登录 ArcUI GridItem 7-zip 卸载 列表 产测工具框架 IMX6ULL 管理框架 Ubuntu共享文件夹 共享目录 Linux共享文件夹 js Linux的基础指令 信号 内核 sdkman curl wget DBeaver 底层实现 单元测试 DeepSeek行业应用 Heroku 网站部署 lsb_release /etc/issue /proc/version uname -r 查看ubuntu版本 PyQt PySide6 springboot远程调试 java项目远程debug docker远程debug java项目远程调试 springboot远程 Web服务器 多线程下载工具 PYTHON 终端 游戏机 即时通信 NIO SWAT 配置文件 服务管理 网络共享 系统开发 binder 车载系统 framework 源码环境 技术 半虚拟化 硬件虚拟化 Hypervisor iNode Macos lio-sam SLAM 视觉检测 micropython esp32 mqtt uv HarmonyOS OpenHarmony Node-Red 编程工具 流编程 小游戏 五子棋 算力 授时服务 北斗授时 服务器管理 宝塔面板 配置教程 网站管理 ubuntu24.04.1 TCP服务器 qt项目 qt项目实战 qt教程 生活 自学笔记 小米 澎湃OS Android 图形化界面 国标28181 监控接入 语音广播 流程 SIP SDP 多进程 网络文件系统 CLion 图片增强 增强数据 linux安装配置 firewalld prometheus数据采集 prometheus数据模型 prometheus特点 社交电子 工业4.0 rnn 高效远程协作 TrustViewer体验 跨设备操作便利 智能远程控制 CentOS Stream CentOS hibernate systemctl composer 安全架构 网络攻击模型 perl workflow W5500 OLED u8g2 SEO mybatis wireshark 小智 chromium dpi 同步 备份 数据采集 Crawlee Playwright vscode 1.86 Cookie 直流充电桩 充电桩 微信开放平台 微信公众平台 微信公众号配置 junit 开机自启动 ping++ Bug解决 Qt platform OpenCV WireGuard 异地组网 fpga embedding Webserver 异步 rpa 考研 Portainer搭建 Portainer使用 Portainer使用详解 Portainer详解 Portainer portainer 在线office 玩游戏 Xinference 网络穿透 自动化测试 设备树 Nuxt.js 命令行 基础入门 ShenTong wsl2 恒源云 yum源切换 更换国内yum源 音乐服务器 Navidrome 音流 servlet springcloud 上传视频至服务器代码 vue3批量上传多个视频并预览 如何实现将本地视频上传到网页 element plu视频上传 ant design vue vue3本地上传视频及预览移除 玩机技巧 软件分享 软件图标 网络原理 okhttp CORS 跨域 分子对接 autodock mgltools PDB PubChem ebpf uprobe 飞牛nas fnos Docker快速入门 harmonyosnext chrome 浏览器下载 chrome 下载安装 谷歌浏览器下载 beautifulsoup 健康医疗 OpenManus 边缘计算 版本 rclone AList fnOS 能力提升 面试宝典 IT信息化 DeepSeek r1 Open WebUI minicom 串口调试工具 向日葵 cd 目录切换 ELF加载 Kylin OS 飞牛NAS 飞牛OS MacBook Pro 程序化交易 量化交易 高频交易 终端工具 远程工具 邮件APP 免费软件 移动云 CDN 计算机 监控 安防软件 mq Ubuntu Server Ubuntu 22.04.5 x64 SIGSEGV xmm0 code-server mosquitto nfs 网页设计 强化学习 WINCC Xshell safari 办公自动化 pdf教程 AISphereButler ollama下载加速 xpath定位元素 llamafactory 微调 Qwen 大数据平台 服务器ssl异常解决 概率论 浏览器自动化 分布式账本 共识算法 linux 命令 sed 命令 云桌面 微软 AD域控 证书服务器 AI作画 自动化任务管理 hosts hosts文件管理工具 开源软件 RagFlow RAG tensorflow Pyppeteer c/c++ 串口 黑客技术 CUPS 打印机 Qt5 pyqt ux 鲲鹏 npu 文件存储服务器组件 fd 文件描述符 client-go k8s二次开发 AI写作 prompt RTMP 应用层 OD机试真题 服务器能耗统计 LVM lvresize 磁盘扩容 pvcreate opensearch helm 模拟实现 蓝耘科技 元生代平台工作流 ComfyUI 服务器主板 AI芯片 DenseNet SenseVoice 代码调试 ipdb top Linux top top命令详解 top命令重点 top常用参数 shard 智能音箱 智能家居 文件系统 路径解析 动态规划 IPMITOOL BMC 硬件管理 NVM Node Yarn PM2 opcua opcda KEPServer安装 打不开xxx软件 无法检查其是否包含恶意软件 计算机系统 RAGFlow mvc string模拟实现 深拷贝 浅拷贝 经典的string类问题 三个swap 游戏服务器 TrinityCore 魔兽世界 rime apt SVN Server tortoise svn Tabs组件 TabContent TabBar TabsController 导航页签栏 滚动导航栏 群晖 文件分享 docker命令大全 pppoe radius 雨云服务器 MQTT协议 消息服务器 代码 token sas dba g++ g++13 rtc 服务器部署ai模型 相机 机柜 1U 2U DIFY 远程看看 远程协助 RoboVLM 通用机器人策略 VLA设计哲学 vlm fot robot 视觉语言动作模型 医疗APP开发 app开发 三级等保 服务器审计日志备份 ldap 蓝牙 云耀服务器 大模型面经 大模型学习 deepseek-v3 ktransformers 支持向量机 流式接口 IO less 隐藏目录 管理器 通配符 gunicorn 网工 充电桩平台 充电桩开源平台 EMUI 回退 降级 Unity Dedicated Server Host Client 无头主机 代理服务器 小番茄C盘清理 便捷易用C盘清理工具 小番茄C盘清理的优势尽显何处? 教你深度体验小番茄C盘清理 C盘变红?!不知所措? C盘瘦身后电脑会发生什么变化? java-zookeeper scapy thingsboard 切换root USB网络共享 磁盘清理 MAC 桌面快捷方式 跨平台 ssh远程登录 Linux awk awk函数 awk结构 awk内置变量 awk参数 awk脚本 awk详解 NFC 近场通讯 智能门锁 nac 802.1 portal 免费 Typore 3GPP 卫星通信 树莓派 VNC retry 重试机制 RAGFLOW gaussdb 华为机试 相差8小时 UTC 时间 浏览器开发 AI浏览器 vsxsrv FunASR ASR Ubuntu 24 常用命令 Ubuntu 24 Ubuntu vi 异常处理 webstorm Github加速 Mac上Github加速 Chrome浏览器插件 c/s 框架 etl 环境配置 免密 公钥 私钥 Wi-Fi 大版本升 升级Ubuntu系统 Claude dity make 显示过滤器 Wireshark安装 我的世界 我的世界联机 copilot EasyConnect telnet 远程登录 AI提示词优化 AimRT Ardupilot 跨域请求 加解密 Yakit yaklang word 软链接 硬链接 ajax netlink libnl3 交叉编译 kerberos ArkTs vscode1.86 1.86版本 ssh远程连接 massa sui aptos sei 影刀 k8s资源监控 annotations自动化 自动化监控 监控service 监控jvm docker desktop PPI String Cytoscape CytoHubba docker搭建pg docker搭建pgsql pg授权 postgresql使用 postgresql搭建 wordpress gitee go MNN SSL证书 线程 csrf cfssl 深度优先 并集查找 换根法 树上倍增 小艺 Pura X cmake HistoryServer Spark YARN jobhistory iDRAC R720xd powerbi 宠物 免费学习 宠物领养 宠物平台 CosyVoice asp.net上传文件夹 asp.net上传大文件 .net core断点续传 .net mvc断点续传 强制清理 强制删除 mac废纸篓 华为证书 HarmonyOS认证 华为证书考试 ubuntu安装 linux入门小白 mm-wiki搭建 linux搭建mm-wiki mm-wiki搭建与使用 mm-wiki使用 mm-wiki详解 金仓数据库 2025 征文 数据库平替用金仓 金仓数据库概述 金仓数据库的产品优化提案 playbook 剧本 yaml Ultralytics 可视化 X11 Xming Ark-TS语言 银河麒麟服务器操作系统 系统激活 uni-file-picker 拍摄从相册选择 uni.uploadFile H5上传图片 微信小程序上传图片 Reactor反应堆 视频平台 录像 视频转发 视频流 AI代码编辑器 顽固图标 启动台 创业创新 数据库管理 ranger MySQL8.0 av1 电视盒子 机顶盒ROM 魔百盒刷机 sonoma 自动更新 fiddler 网站搭建 serv00 hdc 容器技术 Python学习 Python编程 软考设计师 中级设计师 SQL 软件设计师 jina 服务网格 istio 磁盘镜像 服务器镜像 服务器实时复制 实时文件备份 banner csrutil mac恢复模式进入方法 恢复模式 seleium chromedriver chrome devtools HarmonyOS NEXT 原生鸿蒙 openvino notepad++ android-studio llama.cpp 银河麒麟桌面操作系统 达梦 DM8 合成模型 扩散模型 图像生成 火绒安全 mybase OpenGL 图形渲染 对比 meld DiffMerge 端口聚合 windows11 HTTP 服务器控制 ESP32 DeepSeek 本地部署AI大模型 System V共享内存 进程通信 思科模拟器 Cisco Masshunter 质谱采集分析软件 使用教程 科研软件 solidworks安装 image SPI sqlite NLP模型 模拟退火算法 软负载 代码托管服务 内存管理 finebi Alist mount 挂载 网盘 Linux的基础开发工具 fast xfce ruby 端口 查看 ss UDP 银河麒麟高级服务器 外接硬盘 Kylin UOS1070e chfs ubuntu 16.04 regedit 开机启动 AD 域管理 Java Applet URL操作 服务器建立 Socket编程 网络文件读取 easyui 运维监控 可用性测试 rsync 内网服务器 内网代理 内网通信 服务器繁忙 备选 调用 示例 bigdata 毕昇JDK MDK 嵌入式开发工具 论文笔记 致远OA OA服务器 服务器磁盘扩容 实时云渲染 云渲染 3D推流 钉钉 IPv6 IPv6测试 IPv6测速 IPv6检测 IPv6查询 pyscenic 生信教程 飞书 uniapp 欧拉系统 抓包工具 archlinux kde plasma gru XCC Lenovo broadcom 轮播图 qt5 客户端开发 pgpool 业界资讯 端口测试 田俊楠 Bandizip Mac解压 Mac压缩 压缩菜单 cpolar 源代码 wsgiref Web 服务器网关接口 SSL 域名 skynet OpenManage 历史版本 下载 bat d3d12 crosstool-ng 配置原理 docker部署翻译组件 docker部署deepl docker搭建deepl java对接deepl 翻译组件使用 mapreduce 定义 核心特点 优缺点 适用场景 wpf ai工具 做raid 装系统 Jellyfin deepseek r1 LORA docker部署Python pxe h.264 Qwen3 qwen3 32b vllm URL SSH 服务 SSH Server OpenSSH Server 项目部署到linux服务器 项目部署过程 open Euler dde 自定义登录信息展示 motd 美化登录 SystemV cpp-httplib 用户缓冲区 GeneCards OMIM TTD 支付 微信支付 开放平台 MI300x 大模型推理 node Charles 能源 cocoapods 免费域名 域名解析 linux子系统 忘记密码 繁忙 解决办法 替代网站 汇总推荐 AI推理 PostgreSQL15数据库 virtualbox Radius ubuntu24 vivado24 崖山数据库 YashanDB 裸机装机 linux磁盘分区 裸机安装linux 裸机安装ubuntu 裸机安装kali 裸机 脚本 Python教程 pycharm安装 Sealos miniconda saltstack 怎么卸载MySQL MySQL怎么卸载干净 MySQL卸载重新安装教程 MySQL5.7卸载 Linux卸载MySQL8.0 如何卸载MySQL教程 MySQL卸载与安装 检索增强生成 文档解析 大模型垂直应用 GIS 遥感 WebGIS Python基础 Python技巧 swift 腾讯云服务器 轻量应用服务器 linux系统入门 linux命令 trea idea 材料工程 swoole #影刀RPA# 阿里云ECS 机架式服务器 1U工控机 国产工控机 模拟器 yashandb Helm k8s集群 无桌面 bootstrap VGG网络 卷积层 池化层 实习 联想开天P90Z装win10 多个客户端访问 IO多路复用 TCP相关API 宕机切换 服务器宕机 qemu libvirt accept 可执行程序 autodl 嵌入式Linux IPC Linux find grep IPv4/IPv6双栈 双栈技术 网路规划设计 ensp综合实验 IPv4过渡IPv6 IPv4与IPv6 工厂方法模式 数据管理 数据治理 数据编织 数据虚拟化 Obsidian Dataview 电视剧收视率分析与可视化平台 Zoertier 内网组网 idm logstash 多端开发 智慧分发 应用生态 鸿蒙OS vue-i18n 国际化多语言 vue2中英文切换详细教程 如何动态加载i18n语言包 把语言json放到服务器调用 前端调用api获取语言配置文件 增强现实 沉浸式体验 技术实现 案例分析 AR GoogLeNet 流程图 mermaid 虚幻引擎 百度云 矩池云 数据下载 数据传输 问题解决 XFS xfs文件系统损坏 I_O error es SoC 星河版 信创 信创终端 中科方德 软件商店 livecd systemtools 商用密码产品体系 risc-v 技能大赛 文件共享 佛山戴尔服务器维修 佛山三水服务器维修 端口开放 MinIO 无法访问wordpess后台 打开网站页面错乱 linux宝塔面板 wordpress更换服务器 计算机学习路线 编程语言选择 稳定性 看门狗 性能调优 安全代理 web开发 NTP服务器 eNSP 网络规划 VLAN 企业网络 匿名FTP 邮件传输代理 SSL支持 chroot监狱技术 #STC8 #STM32 Spring Security 技术共享 Apache Beam 批流统一 案例展示 数据分区 容错机制 网页服务器 web服务器 LLaMA-Factory glm4 comfyui comfyui教程 vpn 子系统 pthread 线性代数 电商平台 SysBench 基准测试 服务器时间 ecm bpm searxng DOIT 四博智联 导航栏 风扇控制软件 Linux环境 备份SQL Server数据库 数据库备份 傲梅企业备份网络版 TCP协议 抗锯齿 postgres Dify重启后重新初始化 cocos2d 3dcoat can 线程池 Minecraft GRUB引导 Linux技巧 dns是什么 如何设置电脑dns dns应该如何设置 医院门诊管理系统 Logstash 日志采集 Dell HPE 联想 浪潮 动态库 GCC编译器 -fPIC -shared 互联网医院 聊天室 lb 协议 VR手套 数据手套 动捕手套 动捕数据手套 惠普服务器 惠普ML310e Gen8 惠普ML310e Gen8V2 dock 加速 muduo PTrade QMT 量化股票 anythingllm open-webui docker国内镜像 飞牛 ceph aac OS OpenSSH Qwen2.5-VL 红黑树封装map和set webview nosql 企业风控系统 互联网反欺诈 DDoS攻击 SQL注入攻击 恶意软件和病毒攻击 游戏开发 NVIDIA 智能问答 向量数据库 Spring AI Milvus 极限编程 xss access blocked 破解 win服务器架设 windows server 实时日志 logs 我的世界服务器搭建 minecraft hugo 查询数据库服务IP地址 SQL Server 移动开发 分布式训练 语法 状态模式 sequoiaDB react native 捆绑 链接 谷歌浏览器 youtube google gmail IO模型 烟雾检测 yolo检测 消防检测 VS Code 根服务器 三次握手 能效分析 AI Agent 字节智能运维 接口优化 deployment daemonset statefulset cronjob hexo 小智AI服务端 xiaozhi TTS LSTM whistle 规格说明书 知行EDI 电子数据交换 知行之桥 EDI 集群管理 IMX317 MIPI H265 VCU 鸿蒙NEXT 解决方案 lua AzureDataStudio Headless Linux 移动魔百盒 网络建设与运维 GRANT REVOKE mcp服务器 client close flink 线程同步 线程互斥 条件变量 机械臂 ubantu post.io 企业邮箱 搭建邮箱 大大通 第三代半导体 碳化硅 mcp协议 go-zero 存储维护 NetApp存储 EMC存储 超融合 nano Async注解 企业网络规划 华为eNSP 网络库 学习路线 静态IP brew navicat tidb 手机 制造 Python 视频爬取教程 Python 视频爬取 Python 视频教程 Qt QModbus bert Anolis nginx安装 linux插件下载 僵尸世界大战 游戏服务器搭建 漏洞报告生成 Web3 Telegram LVS 接口返回 环境 非root CPU架构 服务器cpu yum换源 蜂窝网络 频率复用 射频单元 无线协议接口RAN 主同步信号PSS paddle docker search Multi-Agent 架构与原理 figma IDEA VM虚拟机 triton 模型分析 MLLMs VLM gpt-4v 物理地址 页表 虚拟地址 粘包问题 vmware tools GPUGEEK easyTier 组网 搭建个人相关服务器 midjourney 青少年编程 sentinel lvs Modbus TCP 动态域名 finalsheel proxy模式 烟花代码 烟花 元旦 本地知识库部署 DeepSeek R1 模型 Putty 花生壳 设计规范 干货分享 黑客工具 密码爆破 RK3568 网络搭建 神州数码 神州数码云平台 云平台 查看显卡进程 fuser ArtTS deekseek UFW vue在线预览excel和编辑 vue2打开解析xls电子表格 浏览器新开页签或弹框内加载预览 文件url地址或接口二进制文档 解决网页打不开白屏报错问题 win向maOS迁移数据 web环境 带外管理 tailscale derp derper 中转 流量运营 MacOS C++软件实战问题排查经验分享 0xfeeefeee 0xcdcdcdcd 动态库加载失败 程序启动失败 程序运行权限 标准用户权限与管理员权限 ISO镜像作为本地源 qtcreator 事件驱动 输入系统 Erlang OTP gen_server 热代码交换 事务语义 homeassistant 欧标 OCPP devmem 多路转接 Ubuntu 24.04.1 轻量级服务器 音乐库 模板 泛型编程 AI员工 项目部署 watchtower AI agent isaacgym 计算虚拟化 弹性裸金属 流量 集成 搜狗输入法 Ubuntu 24.04 搜狗输入法闪屏 Ubuntu中文输入法 fabric orbslam2 anonymous GKI KMI HAProxy ABAP 4 - 分布式通信、分布式张量 华为鸿蒙系统 ArkTS语言 Component 生命周期 条件渲染 Image图片组件 超级终端 多任务操作 提高工作效率 macOS 安装部署 milvus安装 北亚数据恢复 oracle数据恢复 java毕业设计 微信小程序医院预约挂号 医院预约 医院预约挂号 小程序挂号 lrzsz tar.gz tar.xz linux压缩 李心怡 RockyLinux authing ardunio BLE 数字比特流 模拟信号 将二进制数据映射到模拟波形上 频谱资源 振幅频率相位 载波高频正弦波 mujoco RAGflow 中文输入法 西门子PLC 通讯 AWS 命令模式 Metastore Catalog 风扇散热策略 曙光 海光 宁畅 中科可控 jdk11安装 jdk安装 openjdk11 openjdk11安装 零售 wifi驱动 进程池实现 solr pythonai PlaywrightMCP 代码复审 元服务 应用上架 数字证书 签署证书 火山引擎 h.265 STL vCenter服务器 ESXi主机 监控与管理 故障排除 日志记录 caddy qwen2vl nginx默认共享目录 服务器正确解析请求体 容器化 Serverless Unity插件 janus rtcp BCLinux CAD瓦片化 栅格瓦片 矢量瓦片 Web可视化 DWG解析 金字塔模型 nmcli 网络配置 华为昇腾910b3 基本指令 issue socket appium 静态NAT 通用环境搭建 MySQL 站群 多IP 苹果电脑装windows系统 mac安装windows系统 mac装双系统 macbook安装win10双 mac安装win10双系统 苹果电脑上安装双系统 mac air安装win scrapy 影视app AppLinking 应用间跳转 证书 签名 python高级编程 Ansible elk stack sse_starlette Starlette Server-Sent Eve 服务器推送事件 ANDROID_HOME zshrc IP地址 计算机基础 互联网实用编程指南 5分钟快速学 docker入门 MCP 服务器 JADX-AI 插件 Linux的进程概念