多年来,链接被认为是相当直接和无趣的。然而,随着共享库和动态链接在现代操作系统中的重要性越来越高,链接是一个复杂的过程,它为有知识的程序员提供了强大的功能。...例如,许多软件产品在运行时使用共享库来升级压缩打包的二进制文件。此外,许多web服务器依赖于共享库的动态链接来提供动态内容。 2.链接器是啥呢?...链接器:又译为链接器、连结器,是一个程序,将一个或多个由编译器或汇编器生成的目标文件外加库,链接为一个可执行文件。...链接器的作用:简单的讲,链接器的工作就是解析未定义的符号引用,将目标文件中的占位符替换为符号的地址。链接器还要完成程序中各目标文件的地址空间的组织,这可能涉及重定位工作。...这里还会有个新名词,什么是重定位呢? To:重定位,就是将每个符号和内存中的一个位置关联起来,然后修改代码中所有对这些符号的引用,使它们指向这个内存位置。
链接器会解析目标文件之间的引用关系,将它们合并成一个完整的可执行文件。在链接过程中,还会进行地址分配、重定位和符号解析等操作。 静态链接与动态链接 完成两个任务:符号解析与重定位。...符号解析: 建立符号引用和定义之间的联系。 重定位: 为每一个引用确定地址。 链接时间:编译时、加载时、运行时。...共享的目标文件 (.so) 特殊的可重定位目标文件,能在装入或运行时被装入到内存并自动被链接,称为共享库文件,Windows 中称其为 Dynamic Link Libraries (DLLs)。...静态库链接 静态库 定义 将相关可重定位目标模块打包成一个单独的文件。...重定位 重定位由两步组成:重定位节和符号定义,重定位节中的符号引用。 重定位节和符号定义 赋予指令和全局变量唯一的运行时内存地址。
解决办法: 相关的函数被编译为独立的目标模块,然后封成一个单独的静态库文件,然后应用程序可以通过在命令行指定单独的文件名字来使用这些在库中定义的函数。...在Linux系统中,静态库以一种称为存档的特殊文件格式存放在磁盘中,存档文件是一组连续的可重定位目标文件的集合,有一个头部用来描述每个成员目标文件的大小和位置,存档文件名由后缀.a标识。...---- 静态库解析过程 在符号解析阶段,链接器从左到右按照他们在编译器驱动程序命令行上出现的顺序来扫描可重定位目标文件和存档文件(驱动程序自动将命令行中所有.c文件翻译为.o文件)。...在加载时,加载器将部分链接的可执行文件映射到内存,然后调用动态链接器,它通过加载共享库和重定位程序中的引用来完成链接任务。...被编译为位置无关代码的共享库可以加载到任何地方,也可以在运行时被多个进程共享。为了加载、链接和访问共享库的函数和数据,应用程序也可以在运行时使用动态链接器。
链接器主要完成符号解析和重定位两个任务。 目标文件有三种形式:可重定位目标文件(.so);可执行目标文件(.exe),共享目标文件(.so)。....symtab:符号表,存放定义和引用的函数与全局变量的信息。使用 STRIP 命令可以去掉符号表。 .rel.text:.text 中位置的列表,是重定位信息。...链接器把目标文件组合起来的时候,需要修改这些位置,以让各个目标文件链接起来。一般来说,修改的是外部函数或者引用全局变量的位置,调用的本地函数的位置则不需要修改。....rel.data: 引用或定义的所有全局变量的重定位信息。 .debug:调试符号表。用 -g 选项编译的时候才会得到这张表。 .line:源程序的行号与 .text 机器码的对应关系。...静态库用于共享重复的代码,链接器仅会拷贝需要的函数。也可以通过参数拷贝所有函数。 gcc 的静态库链接是按顺序进行的。
静态用.a为后缀, 例如: libhello.a 共享库(动态库)的代码是在可执行程序运行时才载入内存的,在编译过程中仅简单的引用,因此生成的可执行程序代码体积较小。...,其中 xxxx 是该lib的名称; 动态库的名字一般为libxxxx.so.major.minor,xxxx 是该lib的名称,major是主版本号,minor是副版本号 1.5、可执行程序在执行的时候如何定位共享库...; -fPIC:表示编译为位置独立(地址无关)的代码,不用此选项的话,编译后的代码是位置相关的,所以动态载入时,是通过代码拷贝的方式来满足不同进程的需要,而不能达到真正代码段共享的目的。...一、静态库解析符号引用: 链接器ld是如何使用静态库来解析引用的。在符号解析阶段,链接器从左至右,依次扫描可重定位目标文件(*.o)和静态库(*.a)。...在这个过程中,链接器将维持三个集合: 集合E:可重定位目标文件(*.o文件)的集合。 集合U:未解析(未定义)的符号集,即符号表中UNDEF的符号。
优化:编译器会进行一系列的优化操作,以提高生成的机器代码的性能。这包括代码消除、循环展开、内联函数等优化技术。 目标代码生成:目标代码生成阶段将中间代码或其他中间表示翻译为特定体系结构的机器代码。...重定位(Relocation): 目标文件和库文件通常会包含相对于文件起始位置的相对地址,这些地址需要在最终可执行文件中被映射到正确的内存地址上。...动态链接可能发生在两个时机: 加载时的动态链接:操作系统会在执行可执行文件之前,将所需的共享库加载到内存中。这时,链接器会解析可执行文件中的引用,将这些引用关联到所加载的共享库中的实际函数和变量。...运行时的动态链接:共享库已经在加载时加载到了内存中,但链接的最终步骤是在程序运行时进行的。这时,操作系统会确保程序可以正确地访问所需的共享库中的函数和变量。...生成可执行文件(Executable File Generation): 在完成所有的符号解析、重定位和库依赖解析后,链接器会根据上述步骤的结果生成最终的可执行文件。
与弱符号对应的还有弱引用,如果弱引用的符号有定义,链接器决议该符号,如果弱引用的符号未定义,链接器不认为是一个错误。...链接器扫描完所有的重定位表,所有的重定位入口符号都能在全局符号表中找到,否则链接器就会报符号未定义错误。...装载时重定位 共享对象并非完全能被多个进程复用(参照上面共享对象实现的图),一般只有指令部分是进程共享的,而数据部分仍然是进程独立的。...和目标文件一样,共享对象数据段中若有绝对地址引用,会生成对应的重定位表,当动态链接器把这个共享对象装载后,会根据重定位表将数据段中的地址引用修正。这个方法叫做 装载时重定位 。...这些眼熟的表名字实际上功能结构和静态链接时那些表非常相似。最大的区别就是目标文件的重定位是在静态链接时完成,共享对象的重定位是在装载时完成。 值得提出的是可执行文件也可以编译为共享对象形式。
还有,大多数Web服共享库的动牡动太内 静态链接 像 Linux LD程序这样的静态链接器以一组可重定位目标文件和命令行参数作为输入,生成一个完全链接的、可以加载和运行的可执行目标文件作为输出。...符号和符号表 每个可重定位目标模块m都有一个符号表,它包含m定义和引用的符号的信息。在链接器的上下文中,有三种不同的符号: 由模块m定义并能被其他模块引用的全局符号。...共享目标文件(共享库)是在运行时由动态链接器链接和加载的,或者隐含地在调用程序被加载和开始执行时,或者根据需要在程序调用dlopen库的函数时。 ...在加载时,加载器将部分链接的可执行文件映射到内存,然后调用动态链接器,它通过加载共享库和重定位程序中的引用来完成链接任务。...被编译为位置无关代码的共享库可以加载到任何地方,也可以在运行时被多个进程共享。为了加载链接和访问共享库的函数和数据,应用程序也可以在运行时使用动态链接器。
使用objdump的-d参数可以看到“a.o”的代码反汇编结果: objdump -d a.o 我们知道在程序的代码里面使用的都是虚拟地址,在这里也可以看到“main”的起始地址以0x00000000...在没有重定位之前,相对偏移被置为0xFFFFFFFC(小端),它是常量“-4”的补码形式。 让我们来仔细看看这条指令的含义。...重定位的过程中,每个重定位的入口都是对一个符号的引用,那么当链接器须要对某个符号的引用进行重定位时,它就要确定这个符号的目标地址。...比如我们查看“a.o”的符号表: GLOBAL”类型的符号,除了“main”函数是定义在代码段之外,其他两个“ shared和“swap”都是“UND”,即“ undefined”未定义类型,这种未定义的符号都是因为该目标文件中有关于它们的重定位项...所以在链接器扫描完所有的输入目标文件之后,所有这些未定义的符号都应该能够在全局符号表中找到,否则链接器就报符号未定义错误。
小心两个共享库共用同一个静态库.pdf 注:以下内容仅针对Linux/GCC环境,不涵盖Windows,包括Cygwin环境。...原因是在使用dlopen动态加载共享库时,如果静态库中包含有全局变量,可能会出现名同地址不同的全局变量。 解决办法:总是使用RTLD_GLOBAL加载共享库,而不是RTLD_LOCAL。...如果被依赖的不是静态库,而是共享库,则无论何种方式都不存在问题 为何即使RTLD_GLOBAL加载,也会执行两次构造和析构?...原因是两个共享库存在相同的代码段,如果被依赖的是共享库,则不存在这个问题 -Wl的使用 -Wl表示后面的参数传递给链接器,其中l是linker的意思。...表示不丢弃未定义的符号和需要重定位的符号 --export-dynamic 创建一个动态连接的可执行程序时, 把所有的符号加到动态符号表中
、目的档、共享库和核心转储的标准文件格式。...ELF文件主要有四种类型: 可重定位文件(Relocatable File) 包含适合于与其他目标文件链接来创建可执行文件或者共享目标文件的代码和数据。...链接器在处理目标文件时,需要对目标文件中的某些部位进行重定位,即代码段和数据中中那些绝对地址引用的位置。对于每个需要重定位的代码段或数据段,都会有一个相应的重定位表。...地址无关代码(PIC) 装载时重定位是解决动态模块中有绝对地址引用的方法之一,但是它有一个很大的缺点是指令部分无法在多个进程之间共享,这样就失去了动态链接节省内存的一大优势。...用readelf查看elf文件的动态符号表及它的哈希表。 ? 动态链接重定位表 在动态链接中,导入符号的地址在运行时才确定,所以需要在运行时将这些导入符号的引用修正,即需要重定位。
内存映射段是高效的I/O映射方式,用于装载一个共享的动态内存库。用户可使用系统接口创建共享共享内存,做进程间通信。 堆用于程序运行时动态内存分配,堆是可以上增长的。 数据段–存储全局数据和静态数据。...它接受一个参数,即所需内存的大小(以字节为单位),并返回指向这块内存的指针。 初始化:malloc不会对分配的内存进行初始化,内存中的内容是未定义的,可能是之前的值或者全零,具体取决于操作系统。...如果不对齐,可能导致未定义行为。 安全性:使用定位new时,你需要确保所指定的内存区域足够大,以容纳完整的对象实例,包括可能的内部对齐填充。否则,可能会覆盖周边内存,引发严重错误。...标准库支持:C++标准库提供了一个全局的operator new(void*, std::size_t)重载,它不执行任何实际的内存分配,专门用于定位new表达式。...这个重载是固定的,不能被用户自定义版本替代。
在计算机系统中,目标文件通常有三种形式: 1. 可重定位的目标文件:包含二进制代码和数据,与其他可重定位目标文件合并起来,创建一个可执行目标文件。 2....共享目标文件:一种特殊的可重定位目标文件,即我们通常所说的动(静)态链接库 一个典型的可重定位目标文件如下图所示: 高地址 节头部表 .strtab .line .debug .rel.data .rel.text...下面我们在通过符号表来解释变量的存储。 每个可重定位目标文件都有一个符号表,它包含该文件所定义和引用的符号的信息。在链接器的上下文中,有三种不同的符号: 1....由该文件定义并能被其他模块引用的全局符号。即非静态的C函数和非静态的全局变量,如程序中的a,z,swap。 2. 由其他模块定义并被该文件引用的全局符号。用extern关键字所定义的变量和函数。...Vis 目前还没有查到资料,待以后改正 Ndx 通过索引来表示每个节 ABS:不该被重定位的符号 UND:代表未定义的符号(在其他地方定义)
4,ELF文件类型:分为 可重定位、可执行、共享目标文件、核心转储文件 3.4.2 段表(除了文件头以外最重要的结构) 1,描述了ELF各个段的信息(比如每个段的段名、段的长度、在文件中的偏移,读写权限及其他属性...3.4.3 重定位表 1,一个叫”rel.text”的段,类型是”SHT_REL”,就是重定位表 2,链接器在处理目标文件时,需要对目标文件中的某些部位进行重定位(即代码段和数据段中那些对绝对地址的引用的位置...3,这些重定位信息都记录在ELF文件的重定位表里。对于每个需要重定位的代码段或数据段,都会有一个相应的重定位表。...5,弱引用与强引用: 强引用:假如链接时没找到该符号的定义,链接器就会报符号未定义错误。 弱引用:假如链接时没找到该符号的定义,链接器不会报错,默认其为0或是一个特殊值。...7,弱符号和弱引用对库来说非常有用,比如库中定义的弱符号可以被用户定义的强符号所覆盖。
-b :指定目标代码输入文件的格式 -Bstatic:只使用静态库 -Bdynamic:只使用动态库 -Bsymbolic:把引用捆绑到共享库中的全局符号 -c ,--mri-script=:为与MRI链接器兼容,ld接受由MRI命令语言编写的脚本文件 --cref:创建跨引用表 -d,-dc,-dp:即使指定了可重定位的输出文件...这告诉动态链接器,正在创建的共享对象的符号表应该用作共享对象名称的符号表的筛选器。 -g:被忽略。...=:从指定的文件读取符号名称和地址 -r,--relocatable:生成可重定位的输出(称为部分连接) -rpath=:把指定的目录添加到运行时库搜索路径 -rpath-link...:创建共享库 -split-by-file[=size]:为每个目标文件在输出文件中创建额外的段大小达到size。
动态库可以在多个进程间共享,所以动态链接的可执行文件体积更小,节省磁盘空间。操作系统采用的虚拟内存机制允许物理内存中的一份动态库被多个进程共享,节省内存空间。 四、库的本质是什么?...如果我们不想给对方源代码,可以给对方提供.o可重定位目标二进制文件和.h头文件,让对方直接进行链接工作,这样的方式也可以生成可执行程序。 左侧相当于用库的人,右侧相当于写库的人。.../mylib/include/ 2.2 链接错误:函数的未定义引用(库文件找不到,库搜索路径) 1. 指令执行后产生了链接错误,也就是说预处理、编译、汇编阶段已经没有问题了。...下面是使用ncurses库的demo代码,大家也可以在vim上玩一下,在编译代码的时候,要告诉gcc库的名称,否则会报连接错误:函数的未定义引用。...),然后OS会通过页表将物理内存中动态库的位置映射到虚拟地址空间中的共享区,一旦动态库被映射到共享区,那么这个库的起始地址也就立马被确定了,完成映射之后,虚拟地址空间中不是有库函数的偏移量吗?
符号解析与重定位:使用第一步收集到的所有信息,读取输入文件中段的数据及重定位信息,进行符号解析和重定位,调整代码中的地址,将每个段中需要重定位的指令和数据进行“修补”,使他们都指向正确的位置。...对于那些需要重定位的符号,都会放在重定位表里,也叫重定位段,即.rel.data、.rel.text等,如果.text段有被重定位的地方,就有.rel.text段,如果.data段有被重定位的地方,就有...: nm -u test.o U _GLOBAL_OFFSET_TABLE_ U puts 对于UND类型,这种未定义的符号都是因为该目标文件中有关于他们的重定位项...,在链接器扫描完所有的输入目标文件后,所有这种未定义的符号都应该能在全局符号表中找到,否则报符号未定义错误。...I:该符号对另一个符号的间接引用 N:debug符号 R:该符号位于只读数据区 T:该符号位于代码段 U:该符号在当前文件未定义,定义在别的文件中 ?
UNIX环境下主要有三种类型的目标文件: (1)可重定位文件 其中包含有适合于其它目标文件链接来创建一个可执行的或者共享的目标文件的代码和数据。...(2)共享的目标文件 这种文件存放了适合于在两种上下文里链接的代码和数据。...第一种是链接程序可把它与其它可重定位文件及共享的目标文件一起处理来创建另一个 目标文件;第二种是动态链接程序将它与另一个可执行文件及其它的共享目标文件结合到一起,创建一个进程映象。...例如,某个源文件中的函数可能引用了另一个源文件中定义的某个符号(如变量或者函数调用等);在程序中可能调用了某个库文件中的函数,等等。所有的这些问题,都需要经链接程序的处理方能得以解决。...(2) 动态链接 在此种方式下,函数的代码被放到称作是动态链接库或共享对象的某个目标文件中。链接程序此时所作的只是在最终的可执行程序中记录下共享对象的名字以及其它少量的登记信息。
01003 从列函数的参数消去 NULL 值。01004 字符串值在指定给具有较短长度的另一字符串数据类型时被截断。01005 SQLDA 中的条目数不够。01007 未授予特权。...01657 缓冲池操作在下一次数据库重新启动之后才会生效。01665 列名或参数名被截断。01667 可能不能使用视图来优化查询的处理。...42746 在同一类型层次结构中,方法名不能与结构化类型名相同。42748 存储路径对于数据库来说已存在或者被指定了多次。...4274B 在 XML 模式存储库中未找到唯一的 XSROBJECT。4274F 在安全标号组件中未定义组件元素。4274G 在给定安全标号所使用的安全标号策略中未定义安全标号组件。...4274B 在 XML 模式存储库中未找到唯一的 XSROBJECT。 4274F 在安全标号组件中未定义组件元素。 4274G 在给定安全标号所使用的安全标号策略中未定义安全标号组件。
在Linux中静态库以.a为后缀,首先静态库的实现是在文件编译后的链接过程中就被载入到了程序中,那么后续的程序的运行使用就不会再依赖库,那么弊端就是将静态库载入到程序中会导致程序的体积变大,而且如果静态库后续进行了更新...而共享库以.so为后缀,它的实现方式是在链接的过程中不会将库载入到程序中,而是生成一个记录表,当程序运行的时候再去通过记录表去链接共享库,那么这个的好处就是程序的体积小,而且当有不同的程序公用一个库的时候...首先我们先将这些函数文件编译为.o文件,需要加上-fPIC参数,表示让生成的.o文件与路径无关,这一点主要是用于共享库的操作,单纯只用静态库的话可以不加。 ?...首先我们要对.cpp文件进行编译,那么-fPIC这个命令此时就不能少,表示编译的文件与路径无关,也就是说生成的文件都是相对路径,而不是绝对路径,如果使用绝对路径,那么在多个进程共享使用这个库的时候,会根据这个代码段引用的数据对象需要重定位..., 重定位会修改代码段的内容,这就造成每个使用这个.so文件代码段的进程在内核里都会生成这个.so文件代码段的copy,那么就相当于每个进程都维护了一个自己的.so的副本,因此就做不到共享了。
领取专属 10元无门槛券
手把手带您无忧上云