最新资讯

  • Linux——动静态库

Linux——动静态库

2025-05-04 07:00:38 0 阅读

目录

1. 动静态库基本原理

2. 认识动静态库 

3. 动静态库的特点 

3.1 静态库的优缺点

3.2 动态库的优缺点

4. 静态库的打包和使用

4.1 打包

4.2 使用 

5. 动态库的打包和使用

5.1 打包

5.2 使用 

6. 库的理解与加载

6.1 目标文件

6.2 ELF文件 

6.3 ELF形成到加载

6.3.1 ELF形成可执行

6.3.2 ELF可执行文件加载

6.4 理解链接与加载

6.4.1 静态链接

6.4.2 ELF加载与进程地址空间 

6.5 动态链接与动态库加载

6.5.1 进程如何看待动态库

6.5.2 进程间如何共享库 

6.5.3 动态链接


1. 动静态库基本原理

动静态库的本质是可执行程序的“半成品”。

我们先来介绍可执行程序形成的过程:

step1:预处理

完成头文件展开、去注释、宏替换、条件编译等,最终形成xxx.i文件。

step2:编译

完成词法分析、语法分析、语义分析、符号汇总等,检查无误后将代码翻译成汇编指令,最终形成xxx.s文件。

step3:汇编

将汇编指令转换成二进制指令,最终形成xxx.o文件。

step4:链接

将生成的各个xxx.o文件进行链接,最终形成可执行程序。

上面四个过程,大家可以按照ISO的顺序理解记忆。

例如,用test1.c、test2.c、test3.c、test4.c以及main1.c形成可执行文件,我们需要先得到各个文件的目标文件test1.o、test2.o、test3.o、test4.o以及main1.o,然后再将这写目标文件链接起来,最终形成一个可执行程序。

如果我们在另一个项目当中也需要用到test1.c、test2.c、test3.c、test4.c和项目的main2.c或者main3.c分别形成可执行程序,那么可执行程序生成的步骤也是一样的。

这里大家会发现,我们做了重复的工作,频繁对源文件进行操作,那有没有什么方法可以简化我们的操作呢?这里就要引入“库”的概念了。

这里的test1.c、test2.c、test3.c、test4.c,我们可以将它们的目标文件test1.o、test2.o、test3.o、test4.o进行打包,之后需要用到这四个目标文件时就可以之间链接这个包当中的目标文件了,而这个包实际上就可以称之为一个库。

实际上,所有库本质都是一堆目标文件(xxx.o)的集合,库的文件当中并不包含主函数而只是包含了大量的方法以供调用,所以说动静态库本质是可执行程序的“半成品”

2. 认识动静态库 

这里写了一段很简单的代码,足够让大家认识动静态库。

在这份代码当中我们可以通过调用printf输出Hello Linux,主要原因是gcc编译器在生成可执行程序时,将C标准库也链接进来了

那什么是C标准库,其实我们可以通过命令查看它;

在Linux下,我们可以通过“ldd 文件名”来查看一个可执行程序所依赖的库文件。

这其中的libc.so.6就是该可执行程序所依赖的库文件,我们通过ls命令可以发现libc.so.6实际上只是一个软链接

实际上该软链接的源文件/lib/x86_64-linux-gnu/ld-linux-x86-64.so.2和/lib64/ld-linux-x86-64.so.2在同一个目录下,为了进一步了解,我们可以通过“file 文件名”命令来查看/lib/x86_64-linux-gnu/ld-linux-x86-64.so.2的文件类型。

我们可以看到/lib/x86_64-linux-gnu/ld-linux-x86-64.so.2是一个共享目标文件库(动态库又叫作共享库),后面我们可以看到它是动态链接形成的动态库,当我们去掉一个动静态库的前缀lib,再去掉后缀.so或者.a及其后面的版本号,剩下的就是这个库的名字。

  • 在Linux当中,以.so为后缀的是动态库,以.a为后缀的是静态库。
  • 在Windows当中,以.dll为后缀的是动态库,以.lib为后缀的是静态库。

而gcc/g++编译器默认都是动态链接的,若想进行静态链接,可以携带一个-static选项。

cp@hcss-ecs-348a:~/test1$ gcc -o mytest-s mytest.c -static

此时生成的可执行程序就是静态链接的了,可以明显发现静态链接生成的可执行程序的文件大小,比动态链接生成的可执行程序的文件大小要大得多。

静态链接生成的可执行程序并不依赖其他库文件,此时当我们使用ldd 文件名命令查看该可执行程序所依赖的库文件时就会看到以下信息。

此外,当我们分别查看动静态链接生成的可执行程序的文件类型时,也可以看到它们分别是动态链接和静态链接的。

3. 动静态库的特点 

3.1 静态库的优缺点

静态库是程序在编译链接的时候把库的代码复制到可执行文件当中,生成的可执行程序在运行的时候将不再需要静态库,因此使用静态库生成的可执行程序的大小一般比较大。

优点:

  • 使用静态库生成可执行程序后,该可执行程序就可以独自运行,不再需要库了。

缺点:

  • 使用静态库生成可执行程序会占用大量空间,特别是当有多个静态程序同时加载而这些静态程序使用的都是相同的库,这时在内存当中就会存在大量的重复代码。

3.2 动态库的优缺点

动态库是程序在运行的时候才去链接相应的动态库代码的,多个程序共享使用库的代码。一个与动态库链接的可执行文件仅仅包含它用到的函数入口地址的一个表,而不是外部函数所在目标文件的整个机器码。

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

动态库在多个程序间共享,节省了磁盘空间,操作系统采用虚拟内存机制允许物理内存中的一份动态库被要用到该库的所有进程共用,节省了内存和磁盘空间。

优点:

  • 节省磁盘空间,且多个用到相同动态库的程序同时运行时,库文件会通过进程地址空间进行共享,内存当中不会存在重复代码。
  • 减少页面交换。

缺点:

  • 必须依赖动态库,否则无法运行。
  • 运行加载速度相较静态库慢一些。

4. 静态库的打包和使用

4.1 打包

这里我们先来创建几个头文件和源文件。

add.h

#pragma once
extern int add(int x,int y);

add.c 

#include"add.h"
int add(int x,int y)
{
    return x+y;
}

mul.h 

#pragma once
extern int mul(int x,int y);

mul.c 

#include"mul.h"
int mul(int x,int y)
{
    return x*y;
}

下面我们来进行打包:

第一步:让所有源文件生成对应的目标文件 

第二步:使用ar命令将所有目标文件打包为静态库 

ar命令是gnu的归档工具,常用于将目标文件打包为静态库,下面我们使用ar命令的-r选项和-c选项进行打包。

此外,我们可以用ar命令的-t选项和-v选项查看静态库当中的文件。 

第三步:将头文件和生成的静态库组织起来 

当我们把自己的库给别人用的时候,实际上需要给别人两个文件夹,一个文件夹下面放的是一堆头文件的集合,另一个文件夹下面放的是所有的库文件

因此,在这里我们可以将add.h和mul.h这两个头文件放到一个名为include的目录下,将生成的静态库文件libcal.a放到一个名为lib的目录下,然后将这两个目录都放到mathlib下,此时就可以将mathlib给别人使用了。

打包完成后,我们可以通过tree命令来查看库结构。

4.2 使用 

创建源文件main.c,编写下面这段简单的程序来尝试使用我们打包好的静态库。

#include
#include
int main()
{
    int a=5;
    int b=6;
    int c=mul(a,b);
    printf("%d*%d=%d
",a,b,c);
    return 0;
}

方法1:使用选项 

此时使用gcc编译main.c生成可执行程序时需要携带三个选项:

  • -I:指定头文件搜索路径。
  • -L:指定库文件搜索路径。
  • -l:指明需要链接库文件路径下的哪一个库。
cp@hcss-ecs-348a:~/test2$ gcc main.c -I./mathlib/include -L./mathlib/lib -lcal

这里我们使用自己制作的库生成了可执行程序,并且完成了计算工作。

几点说明:

1.、因为编译器不知道你所包含的头文件add.h在哪里,所以需要指定头文件的搜索路径。
2.、因为头文件add.h当中只有mul函数的声明,并没有该函数的定义,所以还需要指定所要链接库文件的搜索路径。
3、实际中,在库文件的lib目录下可能会有大量的库文件,因此我们需要指明需要链接库文件路径下的哪一个库。库文件名去掉前缀lib,再去掉后缀.so或者.a及其后面的版本号,剩下的就是这个库的名字。
4、-I(大写i),-L,-l(小写l)这三个选项后面可以加空格,也可以不加空格。

到这里,其实我们已经实现了静态库,但是肯定有人会想,编译一个可执行程序还带那么多选项,是不是太麻烦了,能不能简单一点?答案是OK的,大家来看第二种方法:

方法2:把头文件和库文件拷贝到系统路径下

要实现这个方法,首先大家需要知道系统路径在哪里?

所以我们只需要将我们的头文件和库文件拷贝到系统对应路径下即可。

cp@hcss-ecs-348a:~/test1$ sudo cp mathlib/include/*.h /usr/include/
cp@hcss-ecs-348a:~/test1$ sudo cp mathlib/lib/libcal.a /usr/lib/x86_64-linux-gnu

这里我们就只需要说明是哪个库就可以形成可执行程序了。 

5. 动态库的打包和使用

5.1 打包

动态库的打包相对于静态库来说有一点点差别,但大致相同,我们还是利用四个文件进行打包演示:

第一步:让所有源文件生成对应的目标文件 

此时用源文件生成目标文件时需要携带-fPIC选项:

第二步:使用-shared选项将所有目标文件打包为动态库 

第三步:将头文件和生成的动态库组织起来 

与生成静态库时一样,为了方便别人使用,在这里我们可以将add.h和sub.h这两个头文件放到一个名为include的目录下,将生成的动态库文件libcal.so放到一个名为lib的目录下,然后将这两个目录都放到mylib下,此时就可以将mylib给别人使用了。

这里的操作和上面基本一致,大家可以类比学习。

5.2 使用 

与上面一样,我们创建main.c

#include
#include
int main()
{
    int a=10;
    int b=2;
    int c=div(a,b);
    printf("%d/%d=%d
",a,b,c);
    return 0;
}

 

下面还是和静态库那里一样,我们使用选项:

我们使用-I,-L,-l这三个选项都是在编译期间告诉编译器我们使用的头文件和库文件在哪里以及是谁,但是当生成的可执行程序生成后就与编译器没有关系了,相当于我们只告诉了gcc,而没有告诉OS;此后该可执行程序运行起来后,操作系统找不到该可执行程序所依赖的动态库,我们可以使用ldd命令进行查看,如上图所示。

那为啥静态库没有这个问题呢?其实答案前面已经说过了:

静态库是程序在编译链接的时候把库的代码复制到可执行文件当中的。这也就意味着一旦形成可执行程序,它将不再依赖静态库。

那么我们应该怎么解决呢?这里介绍一个比较简单的方法:

既然系统找不到我们的库文件,那么我们直接将库文件拷贝到系统共享的库路径下,这样一来系统就能够找到对应的库文件了。

cp@hcss-ecs-348a:~/test3$ sudo cp mylib/lib/libcal.so /lib/x86_64-linux-gnu

最后再总结一些结论 

1. gcc/g++默认使用动态库(动静态库同时存在时)。如果非要静态链接,就要带-static,而且必须有静态库的存在。 如果只存在静态库,那么对于该库就只能静态链接了。

2. Linux中,默认安装的大部分都是动态库。

3. 库:应用程序=1:n

6. 库的理解与加载

6.1 目标文件

⽬标⽂件(.o文件)是⼀个⼆进制的文件,文件的格式是 ELF ,是对二进制代码的⼀种封装。

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

tip:动静态库、可执行程序、.o文件都是ELF格式。 

6.2 ELF文件 

• 可重定位文件(Relocatable File) :即xxx.o文件。包含适合于与其他⽬标文件链接来创建可执行文件或者共享⽬标文件的代码和数据。

• 可执行文件(Executable File) :即可执行程序。

• 共享⽬标文件(Shared Object File) :即xxx.so⽂件。

• 内核转储(core dumps) ,存放当前进程的执行上下文,用于dump信号触发。

⼀个ELF文件由以下四部分组成:

• ELF头(ELF header) :描述⽂件的主要特性。其位于⽂件的开始位置,它的主要⽬的是定位⽂件的其他部分。

• 程序头表(Program header table) :列举了所有有效的段(segments)和他们的属性。表⾥记着每个段的开始的位置和位移(offset)、长度,毕竟这些段,都是紧密的放在⼆进制文件中, 需要段表的描述信息,才能把他们每个段分割开。

• 节头表(Section header table) :包含对节(sections)的描述。

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

6.3 ELF形成到加载

6.3.1 ELF形成可执行

• step-1:将多份 C/C++ 源代码,翻译成为⽬标 .o 文件。

• step-2:将多份 .o ⽂件section进行合并。

注意: • 实际合并是在链接时进行的,但是并不是这么简单的合并,也会涉及对库合并,此处不做 过多追究。

6.3.2 ELF可执行文件加载

• ⼀个ELF会有多种不同的Section,在加载到内存的时候,也会进⾏Section合并,形成segment

• 合并原则:相同属性,⽐如:可读,可写,可执⾏,需要加载时申请空间等.

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

• 很显然,这个合并工作也已经在形成ELF的时候,合并⽅式已经确定了,具体合并原则被记录在了 ELF的 程序头表(Program header table) 中。

# 查看可执⾏程序的section 
$ readelf -S a.out 
There are 31 section headers, starting at offset 0x19d8:
Section Headers:
 [Nr] Name Type Address Offset
 Size EntSize Flags Link Info Align
[ 0] NULL 0000000000000000 00000000
 0000000000000000 0000000000000000 0 0 0
 [ 1] .interp PROGBITS 0000000000400238 00000238
 000000000000001c 0000000000000000 A 0 0 1
 [ 2] .note.ABI-tag NOTE 0000000000400254 00000254
 0000000000000020 0000000000000000 A 0 0 4
 [ 3] .note.gnu.build-i NOTE 0000000000400274 00000274
 0000000000000024 0000000000000000 A 0 0 4
 [ 4] .gnu.hash GNU_HASH 0000000000400298 00000298
 000000000000001c 0000000000000000 A 5 0 8
 [ 5] .dynsym DYNSYM 00000000004002b8 000002b8
 0000000000000048 0000000000000018 A 6 1 8
 [ 6] .dynstr STRTAB 0000000000400300 00000300
 0000000000000038 0000000000000000 A 0 0 1
 [ 7] .gnu.version VERSYM 0000000000400338 00000338
 0000000000000006 0000000000000002 A 5 0 2
 [ 8] .gnu.version_r VERNEED 0000000000400340 00000340
 0000000000000020 0000000000000000 A 6 1 8
 [ 9] .rela.dyn RELA 0000000000400360 00000360
 0000000000000018 0000000000000018 A 5 0 8
 [10] .rela.plt RELA 0000000000400378 00000378
 0000000000000018 0000000000000018 AI 5 24 8
 [11] .init PROGBITS 0000000000400390 00000390
 ...
# 查看section合并的segment 
$ readelf -l a.out 
Elf file type is EXEC (Executable file)
Entry point 0x4003e0
There are 9 program headers, starting at offset 64
Program Headers:
 Type                  Offset                   VirtAddr                   PhysAddr
 FileSiz MemSiz Flags Align
 PHDR               0x0000000000000040     0x0000000000400040         0x0000000000400040
                    0x00000000000001f8     0x00000000000001f8                R E 8
 INTERP             0x0000000000000238     0x0000000000400238         0x0000000000400238
                    0x000000000000001c     0x000000000000001c                R    1
 [Requesting program interpreter: /lib64/ld-linux-x86-64.so.2]
 LOAD               0x0000000000000000     0x0000000000400000         0x0000000000400000
                    0x0000000000000744     0x0000000000000744         R E 200000
 LOAD               0x0000000000000e10     0x0000000000600e10         0x0000000000600e10
                    0x0000000000000218     0x0000000000000220         RW 200000
 DYNAMIC            0x0000000000000e28     0x0000000000600e28        0x0000000000600e28
                    0x00000000000001d0     0x00000000000001d0         RW 8
 NOTE               0x0000000000000254     0x0000000000400254         0x0000000000400254
                    0x0000000000000044     0x0000000000000044         R 4
 GNU_EH_FRAME       0x00000000000005a0     0x00000000004005a0         0x00000000004005a0
                    0x000000000000004c     0x000000000000004c         R 4
 GNU_STACK          0x0000000000000000     0x0000000000000000         0x0000000000000000
                    0x0000000000000000     0x0000000000000000         RW 10
 GNU_RELRO          0x0000000000000e10     0x0000000000600e10         0x0000000000600e10
                    0x00000000000001f0     0x00000000000001f0         R 1
 Section to Segment mapping:
 Segment Sections...
 00 
 01 .interp 
 02 .interp .note.ABI-tag .note.gnu.build-id .gnu.hash .dynsym .dynstr 
.gnu.version .gnu.version_r .rela.dyn .rela.plt .init .plt .plt.got .text 
.fini .rodata .eh_frame_hdr .eh_frame 
 03 .init_array .fini_array .jcr .dynamic .got .got.plt .data .bss 
 04 .dynamic 
 05 .note.ABI-tag .note.gnu.build-id 
 06 .eh_frame_hdr 
 07 
 08 .init_array .fini_array .jcr .dynamic .got 

为什么要将section合并成为segment?

• Section合并的主要原因是为了减少⻚⾯碎⽚,提⾼内存使⽤效率。如果不进行合并, 假设页面大小为4096字节(内存块基本⼤小,加载,管理的基本单位),如果.text部分为4097字节,.init部分为512字节,那么它们将占⽤3个⻚⾯,⽽合并后,它们只需2个页面。

• 此外,操作系统在加载程序时,会将具有相同属性的section合并成⼀个大的 segment,这样就可以实现不同的访问权限,从⽽优化内存管理和权限访问控制。 

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

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

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

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

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

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

从链接视图来看

• 命令 readelf -S hello.o 可以帮助查看ELF⽂件的节头表。

• .text节 :是保存了程序代码指令的代码节。

• .data节 :保存了初始化的全局变量和局部静态变量等数据。

• .rodata节 :保存了只读的数据,如⼀⾏C语⾔代码中的字符串。由于.rodata节是只读的,所以只能存在于⼀个可执行⽂件的只读段中。因此,只能是在text段(不是data段)中找到.rodata 节。

• .BSS节 :为未初始化的全局变量和局部静态变量预留位置

• .symtab节 :Symbol Table符号表,就是源码⾥⾯那些函数名、变量名和代码的对应关系。 

• .got.plt节 (全局偏移表-过程链接表):.got节保存了全局偏移表。.got节和.plt节⼀起提供了对导入的共享库函数的访问入口,由动态链接器在运行时进行修改。对于GOT的理解,我们后面会说。

◦ 使⽤ readelf 命令查看.so⽂件可以看到该节。

从执行视图来看

• 告诉操作系统哪些模块可以被加载进内存。

• 加载进内存之后哪些分段是可读可写,哪些分段是只读,哪些分段是可执行的。

我们可以在 ELF头中找到文件的基本信息,以及可以看到ELF头是如何定位程序头表和节头表的。

// 查看可执⾏程序 
$ 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 
 Class: ELF64
 Data: 2's complement, little endian
 Version: 1 (current)
 OS/ABI: UNIX - System V
 ABI Version: 0
 Type: DYN (Shared object file)
 Machine: Advanced Micro Devices X86-64
 Version: 0x1
 Entry point address: 0x1060
 Start of program headers: 64 (bytes into file)
 Start of section headers: 14768 (bytes into file)
 Flags: 0x0
 Size of this header: 64 (bytes)
 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

对于 ELF HEADER 这部分来说,我们只⽤知道其作⽤即可,它的主要目的是定位文件的其他部分。 

6.4 理解链接与加载

6.4.1 静态链接

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

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

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

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

6.4.2 ELF加载与进程地址空间 

这里首先来问大家两个问题:

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

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

答案:

• ⼀个ELF程序,在没有被加载到内存的时候,本来就有地址,当代计算机工作的时候,都采⽤"平坦 模式"进行⼯作。所以也要求ELF对自己的代码和数据进行统⼀编址,下⾯是 objdump -S 反汇编之后的代码。

最左侧的就是ELF的虚拟地址,其实,严格意义上应该叫做逻辑地址(起始地址+偏移量),但是我们 认为起始地址是0。也就是说,其实虚拟地址在我们的程序还没有加载到内存的时候,就已经把可执行程序进行统一编址了。

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

从ELF各个 segment来,每个segment有自己的起始地址和自己的长度,用来初始化内核结构中的[start,end]等范围数据,另外在用详细地址,填充页表.。

所以:虚拟地址机制,不光光OS要支持,编译器也要支持。

说到这里,我们需要来重新认识一下虚拟地址空间;

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

 

大家仔细观察上面这张图,当可执行程序加载到内存之前,在磁盘上已经采用“平坦模式”进行统一编址了,在磁盘上叫“逻辑地址”,到了内存里就叫“虚拟地址”,本质上的一个东西。那么可执行程序加载到内存里,其代码和数据本身就占有了对应的物理地址,我们前面学习虚拟地址空间时说过,OS会为我们构建页表来建立虚拟地址和物理地址映射关系;与此同时,在mm_struct中的代码段会被segment中起始地址初始化,也就是确定了代码段的start和end;并且,可执行程序将其入口地址填充到CPU的EIP寄存器中,这样一来CPU就知道了可执行程序从哪里开始执行,然后通过MMU,再加上页表,找到对应程序语句的物理地址,执行语句,如果语句中还在调用其他地方的语句,那么CPU会得到相应的虚拟地址,然后继续查页表,去找到对应物理地址,所以总结来说,进入CPU的都是虚拟地址,出来的都是物理地址。

上图中vm_area_struct中的start和end就是由ELF中的segment初始化的,每个segment加载进来之前就已经经过编址了,所以就用它们上面的起始位置来初始化vm_area_struct。

6.5 动态链接与动态库加载

6.5.1 进程如何看待动态库

6.5.2 进程间如何共享库 

6.5.3 动态链接

首先要交代⼀个结论,动态链接实际上将链接的整个过程推迟到了程序加载的时候。

静态链接最大的问题在于生成的文件体积大,并且相当耗费内存资源。

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

1. 设置堆栈:为程序创建⼀个初始的堆栈环境。

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

3. 动态链接:这是关键的⼀步, _start 函数会调⽤动态链接器的代码来解析和加载程序所依赖的 动态库(shared libraries)。动态链接器会处理所有的符号解析和重定位,确保程序中的函数调 ⽤和变量访问能够正确地映射到动态库中的实际地址。

我们的程序,怎么进行库函数调用? 

• 库已经被我们映射到了当前进程的地址空间中。

• 库的虚拟起始地址我们也已经知道了。

• 库中每⼀个⽅法的偏移量地址我们也知道。

所以:访问库中任意方法,只需要知道库的起始虚拟地址+方法偏移量即可定位库中的方法。

• 而且:整个调用过程,是从代码区跳转到共享区,调用完毕在返回到代码区,整个过程完全在进程地址空间中进行的。

• 也就是说,我们的程序运行之前,先把所有库加载并映射,所有库的起始虚拟地址都应该提前知道。

• 然后对我们加载到内存中的程序的库函数调用进行地址修改,在内存中二次完成地址设置 (这个叫做加载地址重定位)。

• 但是这里就有问题了,修改的是代码区?不是说代码区在进程中是只读的吗?怎么修改?能修改吗?  

这里首先可以明确,代码区肯定不能被修改,这是基本原则,那么这个是怎么实现的呢?

1、动态链接采用的做法是在 .data (可执行程序或者库自己)中专门预留⼀片区域用来存放函数的跳转地址,它也被叫做全局偏移量表GOT,表中每⼀项都是本运行模块要引用的⼀个全局变量或函数 的地址。因为.data区域是可读写的,所以可以支持动态进行修改。 

2、由于代码段只读,我们不能直接修改代码段。但有了GOT表,代码便可以被所有进程共享。但在不同进程的地址空间中,各动态库的绝对地址、相对位置都不同。反映到GOT表上,就是每个进程的 每个动态库都有独立的GOT表,所以进程间不能共享GOT表。

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

4、在调用函数的时候会首先查表,然后根据表中的地址来进行跳转,这些地址在动态库加载的时候会被修改为真正的地址。

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

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

搜索文章

Tags

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