首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

在Linux中构建共享库时如何使链接器因未定义的引用而失败

在Linux中构建共享库时,如果链接器因未定义的引用而失败,可以通过以下步骤解决:

  1. 确定未定义的引用:首先,需要确定哪些符号或函数在链接过程中未定义。可以通过查看链接器的错误输出或者使用工具如nm来检查共享库中的符号表。
  2. 确保所有依赖项已安装:未定义的引用可能是由于缺少依赖项导致的。确保所有需要的库和头文件已正确安装,并且路径设置正确。
  3. 检查编译选项:检查编译共享库时使用的选项,确保所有需要的源文件都包含在编译过程中,并且编译选项正确设置。
  4. 检查链接选项:检查链接共享库时使用的选项,确保所有需要的库都包含在链接过程中,并且链接选项正确设置。
  5. 检查符号可见性:在共享库中,符号的可见性是非常重要的。确保需要在共享库外部使用的符号被正确声明为可见。
  6. 检查版本兼容性:如果共享库依赖于其他库,确保版本兼容性。不同版本的库可能具有不同的符号定义,导致链接失败。
  7. 使用--no-undefined选项:在链接共享库时,可以使用--no-undefined选项告诉链接器不允许未定义的引用。这将使链接器在链接过程中发现未定义的引用并报错。

总结起来,解决链接器因未定义的引用而失败的问题需要仔细检查编译和链接过程中的选项、依赖项、符号可见性和版本兼容性。确保所有需要的文件和库都正确包含,并且路径设置正确。如果仍然无法解决问题,可以尝试使用--no-undefined选项来强制链接器报错并定位未定义的引用。

页面内容是否对你有帮助?
有帮助
没帮助

相关·内容

我与C语言二周目邂逅vlog——8.编译和链接

静态库会在链接时被拷贝到可执行文件中,而动态库则是在程序运行时动态加载的。 6.3 链接的类型 静态链接:在静态链接中,链接器将所有目标文件和所需的库函数全部复制到最终的可执行文件中。...因此,静态链接生成的可执行文件体积较大,但在运行时不再依赖外部库。 动态链接:在动态链接中,链接器只将动态库的引用加入到可执行文件中,而动态库的实际内容则在程序运行时由操作系统加载。...7.3 链接顺序 在使用静态库时,链接的顺序可能会影响最终的链接结果。通常,链接器会按顺序扫描库文件,因此被依赖的库应放在依赖它们的库之后,否则可能出现未定义引用的问题。 8....例如,在 Linux 中,标准库的静态库为 libc.a。 动态链接库(.so 文件):动态链接库在程序运行时被加载,多个程序可以共享一个动态链接库,从而节省内存和磁盘空间。...例如,在 Linux 中,标准库的动态库为 libc.so。 9.3 链接器脚本 链接器脚本(Linker Script)是链接器的配置文件,用于控制链接的方式和最终可执行文件的布局。

11610

关于protobuf近期版本(v20v3.20+)和 gRPC v1.54版本在某些编译环境下的一些链接和编译问题

这就意味着编译 protobuf 的时候是可能被优化掉而没有这个符号的。 但是使用者认为有这个符号,最终链接失败。...触发条件比较多: 需要编译成动态库 默认符号隐藏(Windows默认隐藏,Linux默认可见) 使用 dllexport_decl= 来设置导出符号 在Windows中个,每一个dll和exec都有自己的符号表和堆管理...而在Linux里,默认是共享且全局可见的。而很多构建系统中会把Windows版本依赖使用静态库,所以很多同学不会碰到这些问题。...在 protobuf 生成的代码中,由于 .pb.cc 中存在全局变量,我们也不能允许同一个全局变量在多个动态库中,否则会重复注册和执行构造析构函数。...有兴趣的小伙伴也可以跟进。 gRPC 的链接和编译问题 gRPC 的 v1.54.0 的链接符号问题 我们在使用高版本编译器时,会尽可能使用高版本的STD标准。

1.6K20
  • C++系列:链接器是如何工作的

    构建大型应用经常会碰到链接器错误,这些错误发生的原因一般有:模块缺失、类库缺失、类库版本不兼容等。...Linux链接器在解析符号引用时所作的决策会静默地影响程序的正确性。比如,在默认情况下,如果错误地定义了多个全局变量,链接器是不会报错的。但是生成的程序会表现出令人困惑的行为,且这种程序是很难调试的。...然而,随着共享库和动态链接在现代操作系统中的重要性越来越高,链接是一个复杂的过程,它为有知识的程序员提供了强大的功能。例如,许多软件产品在运行时使用共享库来升级压缩打包的二进制文件。...链接器的作用:简单的讲,链接器的工作就是解析未定义的符号引用,将目标文件中的占位符替换为符号的地址。链接器还要完成程序中各目标文件的地址空间的组织,这可能涉及重定位工作。...To:重定位,就是将每个符号和内存中的一个位置关联起来,然后修改代码中所有对这些符号的引用,使它们指向这个内存位置。 一般来说,现代操作系统包括静态链接和动态链接。

    1.8K40

    《程序员的自我修养》笔记

    所以在链接器扫描完所有的输入目标文件之后,所有这些未定义的符号都应该能够在全局符号表中找到,否则链接器就报符号未定义错误。...那么对于外部和内部符号就可以去用这个地址去访问这快数据 书中原文:静态链接中的第一个步骤,即目标文件在被链接成最终可执行文件时,输入目标文件中的各个段是如何被合并到输出文件中的,链接器如何为它们分配在输出文件中的空间和地址...一旦输入段的最终地址被确定,接下来就可以进行符号的解析与重定位,链接器会把各个输入目标文件中对于外部符号的引用进行解析,把每个段中须重定位的指令和数据进行“修补”,使它们都指向正确的位置。...SO里面会存储完整的动态库符号信息:也就是导出符号表 就是通过编译共享库的时候可以指定编译器参数 打出共享库目标文件(.o)和 共享库链接信息(.so) ,这个so里面会记录共享库中完整的符号信息,这样连接器在查找符号的时候如果发现可以在...这样链接器就可以对foobar的引用做特殊的处理,使它成为一个对动态符号的引用。

    9910

    CSAPP---第七章-链接

    -static参数告诉编译器驱动程序,链接器应该构建一个完全链接的可执行文件,它可以加载到内存并运行,在加载时无需更进一步链接 链接器运行时,它判定main.o引用了addvec.o定义的...所以,无论何时汇编器遇到对最终位置未知的目标引用,它就会生成一个重定位条目,告诉链接器在将目标文件合并成可执行文件时如何修改这个引用。...这里涉及到CSAPP第九章要讲的虚拟内存机制,该章节中会探讨如何实现库的共享 静态库和共享库构造对比如下: 动态链接基本的思路是当创建可执行文件时,静态执行一些链接,然后在程序加载时,动态完成链接过程...链接器还可能生成部分链接的可执行目标文件,这样的文件中有对定义在共享库中的例程和数据的未解析的引用。...在加载时,加载器将部分链接的可执行文件映射到内存,然后调用动态链接器,它通过加载共享库和重定位程序中的引用来完成链接任务。

    92110

    【Linux】解锁线程基本概念和线程控制,步入多线程学习的大门

    而是一整套资源的时候,我们就应该清楚进程创建成本很高。原因就是:创建进程时还需要构建文件描述符,信号表,PCB,页表,等等这就会造成空间和时间的浪费。...与线程有关的函数构成了一个完整的系列,绝大多数函数的名字都是以“pthread_”打头的 要使用这些函数库,要通过引入头文 链接这些线程函数库时要使用编译器命令的“-lpthread...,会报错:(.text+0x1b): undefined reference to main线程未定义,之所以会出错是因为Linux下使用线程需要引用线程库。...编译器并不认识,不是C语言/C++自带的,而是Linux自己创建的原生线程库文件 再加上线程库之后,就可以正常运行了。 可以看见,主线程和新线程是可以同时运行的!...线程可以同时等待不同的I/O操作 4.2.线程的缺点: 健壮性降低 编写多线程需要更全面更深入的考虑,在一个多线程程序里,因时间分配上的细微偏差或者因共享了不该共享的变量而造成不良影响的可能性是很大的

    13310

    从 RUST 库中公开 FFI

    另外,默认情况下 Rust 库使用 crate-type = ["rlib"],而 FFI 因该是 cdylib。...简而言之,它允许其他编程语言,以预期的名称(在我们的例子中是 battery_get_percentage)在编译后的库中查找已声明的函数,而不是编译器生成的名称, 就像 _ZN7battery_get_percentage17h5179a29d7b114f74E...返回参数 在我的例子中,我想向外部公开一些 Rust 的结构,但是由于实现的原因,它们可能包含一些复杂的结构,而强迫最终用户处理这些东西是一个坏主意。...绑定生成 构建完成后,你将得到库文件,你可以将其发布或发送给客户端程序员,使他们更快乐。...附加说明:我发现这个构建脚本在 docs.rs 中构建文档时出现了一些神秘错误,导致构建失败失败。

    1.9K30

    先别急着“用Rust重写”,可能没有说的那么安全

    我们假定开发者是出于善意而移植代码,只是因移植 bug 而将格式错误或 bug 传递给了 FFI,例如指针和缓冲区长度的不正确值。...由于 C/C++ 程序和 Rust 库之间会共享内存,所以对于来自 Rust 库的此类输入的任何不正确处理,都可能在整个程序中引发内存安全错误。...rusTLS 允许客户端创建证书验证器,并在服务器配置间共享这些验证器。为了实现共享,rusTLS 会使用原子引用计数器(Arc)来表示这些验证器,以便在不再引用验证器时自动回收相应的内存。...时间安全:(2)和(3)可能因不正确的函数参数或重复函数调用而导致 use-after-free 和 double-fee 错误。...此外,TLS 库的 C 实现不一定会依靠特定 API 来释放这些对象(及其引用的对象),而可能仅要求客户端使用标准的 free 函数。

    43330

    百度不问我项目,全程基础拷打,真扎心!

    分配内存的大小的计算:使用new操作符申请内存分配时无须指定内存块的大小,编译器会根据类型信息自行计算,而malloc则需要显式地指出所需内存的尺寸。...这是因为在使用delete操作符释放一个指向派生类对象的基类指针时,如果基类的析构函数不是虚函数,那么只会调用基类的析构函数,而不会调用派生类的析构函数,这样就会导致内存泄漏和未定义行为的问题。...通过将析构函数定义为虚函数,可以确保在释放派生类对象时,先调用派生类的析构函数,再调用基类的析构函数,从而避免内存泄漏和未定义行为的问题。...在多线程编程中,如果多个线程同时访问同一个共享资源,可能会发生竞态条件(Race Condition),导致程序的行为出现未定义的情况。为了避免这种情况的发生,可以使用多线程锁来保护共享资源。...COMMIT:提交一个事务,使之生效。 ROLLBACK:回滚一个事务,使之失效。 在MySQL中,事务默认是关闭的,需要通过设置autocommit参数为0来启用事务。

    24110

    Linux命令(65)——ld命令

    -b :指定目标代码输入文件的格式 -Bstatic:只使用静态库 -Bdynamic:只使用动态库 -Bsymbolic:把引用捆绑到共享库中的全局符号 -c 共享库 -split-by-file[=size]:为每个目标文件在输出文件中创建额外的段大小达到size。...此脚本将替换ld的默认链接器脚本(而不是添加到其中),因此脚本必须指定输出文件所需的所有内容。...org>:使用指定的地址作为bss段的起始点 -t,--trace:在处理输入文件时显示它们的名称 -u ,--undefined=:强制指定符号在输出文件中作为未定义符号...warn-once:对于每个未定义的符号只发出一次警告 -warn-section-align:如果为了对齐而改动了输出段地址,则发出警告 --whole-archive:对于指定的存档文件,在存档中包含所有文件

    17.7K13

    一个奇怪的链接问题

    前言 链接是代码生成可执行文件中一个非常重要的过程。我们在使用一些库函数时,有时候需要链接库,有时候又不需要,这是为什么呢?了解一些链接的基本过程,能够帮助我们在编译时解决一些疑难问题。...,提示对‘exp’未定义的引用,并且抛出链接出错。...再次编译运行: gcc -lm -o expTest expTest.c /tmp/ccYT3E65.o:在函数‘main’中: expTest.c:(.text+0x20):对‘exp’未定义的引用...2.什么时候需要链接? 事实上,C编译器总是主动传送libc.a或libc.so给链接器,也就是说,对于使用包含在libc.a或libc.so库中的函数,是不需要在编译时手动链接的。...这个就涉及到链接器的工作原理了,在此只简单说明一下:链接过程中,需要进行符号解析,并且是按照顺序解析;如果库链接在前,就可能出现库中的符号不会被需要,链接器不会把它加到未解析的符号集合中,那么后面引用这个符号的目标文件就不能解析该引用

    1.6K20

    linux动态库和静态库

    静态用.a为后缀, 例如: libhello.a    共享库(动态库)的代码是在可执行程序运行时才载入内存的,在编译过程中仅简单的引用,因此生成的可执行程序代码体积较小。...在Linux下,动态库和静态库同事存在时,gcc/g++的链接程序,默认链接的动态库。...一、静态库解析符号引用: 链接器ld是如何使用静态库来解析引用的。在符号解析阶段,链接器从左至右,依次扫描可重定位目标文件(*.o)和静态库(*.a)。...在这个过程中,链接器将维持三个集合: 集合E:可重定位目标文件(*.o文件)的集合。 集合U:未解析(未定义)的符号集,即符号表中UNDEF的符号。...此时,静态库f中任何不包含在E中的成员目标文件都将丢弃,链接器将继续下一个文件。 3、当所有输入文件完成后,如果U非空,链接器则会报错,否则合并和重定位E中目标文件,构建出可执行文件。

    12.4K20

    深入理解计算机系统(第三版) CSAPP 杂谈,第7章:链接

    链接器主要完成符号解析和重定位两个任务。 目标文件有三种形式:可重定位目标文件(.so);可执行目标文件(.exe),共享目标文件(.so)。...链接器把目标文件组合起来的时候,需要修改这些位置,以让各个目标文件链接起来。一般来说,修改的是外部函数或者引用全局变量的位置,调用的本地函数的位置则不需要修改。...弱全局符号分配在 COMMON section 中,强全局符号分配在 .bss 中。 静态库用于共享重复的代码,链接器仅会拷贝需要的函数。也可以通过参数拷贝所有函数。...遇到目标文件 .o 时会把未定义和已定义的符号保存起来,遇到存档文件 .a 时,除了前面的操作,还会把 .a 的成员符号与未定义的符号比较,把匹配的成员符号对应的 .o 链接起来。...这样的话因为是顺序的,如果把静态库放在前面,则会错过后面目标文件的匹配,从而在链接完所有文件,却还是有未定义符号,结果编译报错。 所以一般做法是静态库文件放在最后。

    1.1K30

    静态链接库和动态链接库的区别

    1、动态库的构造和析构函数机制在Linux中,提供了一个机制:在加载和卸载动态库时,可以编写一些函数,处理一些相应的事物,我们称这些函数为动态库的构造和析构函数,其代码格式如下:void __attribute...); //my_fini为自定义的析构函数名在编译共享库时,不能使用"-nonstartfiles"或"-nostdlib"选项,否则构建与析构函数将不能正常执行(除非你采取一定措施)。...最后运行main.bin的结果同上。4.3、Windows下和Linux下显示加载动态链接库的比较Windows下动态链接库以“.dll”为后缀,而Linux下得动态链接库是以”.so”为后缀的。.../main运行的结果中很容易知道,当Linux静态库和Linux动态库同名时, gcc命令将优先使用动态库。...")))2、在文件里面需要导出的函数前加上:extern "C" DLL_PUBLIC3、Linux下动态库(so)编译时默认不导出,在Makefile中需要添加:-fvisibility=hidden

    8.4K21

    面向 C++ 的现代 CMake 教程(三)

    我们将讨论对象文件的内部结构,如何进行重定位和引用解析,以及它们的用途。我们将讨论最终可执行文件与其组件的区别以及系统如何构建进程映像。 然后,我们将向您介绍各种库——静态库、共享库和共享模块。...共享库在类 Unix 系统上有.so扩展名,在 Windows 上有.dll。 当构建库(静态、共享或共享模块)时,你经常会遇到这个名字链接来表示这个过程。...这是一个旨在作为插件在运行时加载的共享库版本,而不是在编译时与可执行文件链接的东西。共享模块不会随着程序的启动自动加载(像常规共享库那样)。...当链接器遍历二进制文件时,它将执行以下操作: 收集此二进制文件导出的所有未定义符号并将它们存储以供以后使用 尝试使用此二进制文件中定义的符号解决未定义符号(从迄今为止处理的所有二进制文件中收集)...请注意,在一个繁忙的环境中(在共享的测试运行器上),它可能会因调度而产生不利影响。这可以通过下一个选项稍微缓解。

    67000

    从最小依赖角度谈静态库与动态库的选择及配置策略

    通过理论解析、代码示例与对比表格,帮助开发者在项目架构设计阶段作出更合理的决策。1. 前言在构建大型软件系统时,如何有效地管理模块之间的依赖关系是一个长期关注的话题。...在 C++ 开发中,静态库和动态库的选择以及运行时库的配置(/MT 静态链接与 /MD 动态链接)正是决定外部依赖数量的重要因素。...此策略在以下场景中尤为适用:嵌入式系统与便携应用:部署环境有限或对外部库支持较弱时,静态链接可以确保应用独立运行。...重视内存优化与模块共享:undefined对于大型桌面应用或服务器软件,采用 /MD 动态链接可以利用系统中共享的 CRT DLL,降低内存占用,并在 CRT 更新时获得系统级的安全补丁。...然而,需要注意在链接静态库时避免混用 /MT 与 /MD,否则可能导致链接器报错或运行时不稳定。4.

    14410

    译文:开发人员面临的 10个最常见的JavaScript 问题

    用于单页应用程序(SPA) 开发、图形和动画以及服务器端JavaScript平台的强大基于JavaScript的库和框架并不是什么新鲜事。...传统的,与旧浏览器兼容的解决方案是简单地将你对this的引用保存在变量中,然后可以通过闭包继承,例如: 或者,在较新的浏览器中,可以使用bind()方法传入正确的引用: JavaScript 问题#2...浏览器中有一个垃圾回收器,用于清理无法访问的对象占用的内存;换句话说,当且仅当 GC认为对象无法访问时,才会从内存中删除对象。...以下是严格模式的一些关键好处: ·使调试变得更容易。否则将被忽略或默默失败的代码错误现在将生成错误或抛出异常,更快地提醒您代码库中的JavaScript问题,并更快地将您引导到它们的源代码。...如果没有严格模式,对空或未定义的this值的引用会自动强制到全局。这可能会导致许多令人沮丧的错误。在严格模式下,引用this值为null或未定义会引发错误。 ·禁止重复的属性名称或参数值。

    1.3K20

    Kubernetes 中容器的退出状态码参考指南

    以下是容器使用的最常见的退出码: 退出码 名称 含义 0 正常退出 开发者用来表明容器是正常退出 1 应用错误 容器因应用程序错误或镜像规范中的错误引用而停止 125 容器未能运行 docker run...例如,在 Docker 中,尝试 docker start 而不是 docker run; 测试您是否能够使用相同的用户名或上下文在主机上运行其他容器。...与退出码 126 相同,识别失败的命令,并确保容器镜像中引用的文件名或文件路径真实有效。 退出码 128:退出时使用的参数无效 退出码 128 表示容器内的代码触发了退出命令,但没有提供有效的退出码。...可能的原因是: 当通过容器引擎杀死容器时触发,例如使用 docker kill 命令时; 由 Linux 用户向进程发送 kill -9 命令触发; 在尝试终止容器并等待 30 秒的宽限期后由 Kubernetes...: 如果退出代码为 0:容器正常退出,无需排查 如果退出代码在 1-128 之间:容器因内部错误而终止,例如镜像规范中缺少或无效的命令 如果退出代码在 129-255 之间:容器因操作信号而停止,例如

    32810

    【编程基础】C语言内存使用的常见问题

    若变量定义时均初始化,则会产生重定义(multiple definition)的链接错误;若某处变量定义时未初始化,则无链接错误,仅在因类型不同而大小不同时可能产生符号大小变化(size of symbol...在最坏情况下,编译链接正常,但不同文件对同名全局变量读写时相互影响,引发非常诡异的问题。这种风险在使用无法接触源码的第三方库时尤为突出。 【对策】 尽量避免使用全局变量。...一旦链接错误的库,则可能出现某个内存管理器中分配的内存,在另一个内存管理器中释放的问题。...,如显示电脑游戏或动画视频画面时; Ÿ 程序能够请求未被释放的内存(如共享内存),甚至在程序终止时; Ÿ 泄漏发生在操作系统内部或关键驱动中; Ÿ 内存受限,如嵌入式系统或便携设备; Ÿ...但当某次分配失败退出时,未释放系列中其他已成功分配的内存。 7 使用已释放堆内存 动态内存被释放后,其中的数据可能被应用程序或堆分配管理器修改。

    3.4K60
    领券