Linux C/C++ 库链接选项 --whole-archive,--no-whole-archive和--start-group, --end-group

devtools/2024/10/21 11:59:18/

库链接选项

  • 一、介绍
    • whole-archive编译选项
    • Bstatic编译选项
    • start-group 编译选项
  • 示例
  • 参考链接

一、介绍

  这四个都是链接器的选项,所以在编译的时候要用-Wl,[options]来传递给链接器,不然编译器会不认得这个选项。
  在大型工程开发中,工程目录会分的非常细,在modules下会创建很多模块,每个modules都会编译成.a静态库文件供主路径/app/src/下使用,这样就会产生链接库问题,比如在app下的函数要调用静态库里的函数,会出现未定义的错误,需要通过:

-Wl,--whole-archive -lXXX
-Wl,--no-whole-archive -lXXX

  在脚本里修改下链接库的选项,这里XXX代表静态库的名称。

whole-archive编译选项

  默认情况下,对于未使用到的符号(函数是一种符号),链接器不会将它们链接进共享库和可执行程序。这个时候,可以启用链接参数 "--whole-archive"来告诉链接器,将后面库中所有符号都链接进来,参数 "-no-whole-archive" 则是重置,以避免后面库的所有符号被链接进来。

Bstatic编译选项

"-Wl,-Bstatic"指定链接静态库,使用"-Wl,-Bdynamic"指定链接共享库,使用示例:

-Wl,-Bstatic -lmysqlclient_r -lssl -lcrypto -Wl,-Bdynamic -lrt -Wl,-Bdynamic -pthread -Wl,-Bstatic -lgtest

start-group 编译选项

  假设程序x依赖三个静态库:libX1.a、libX2.a和libX3.a,而libX2.a又依赖libX1.a,libX3.a依赖libX2.a和libX1.a,正常情况下的CMakeLists.txt格式如下

target_link_libraries(xlibX1.alibX2.alibX3.a
)

  但也不关心静态库的顺序问题,ld为此提供了start-group和end-group两个选项,让包含在这两者间的静态库顺序可以随意。

target_link_libraries(x-Wl,--start-grouplibX3.alibX2.alibX1.a-Wl,--end-group
)

  这里有一个链接顺序的问题,在command line上前面的库会依赖之后的库,如果碰到循环链接,-lliba -llibb -lliba,这样就需要使用–start-group和–end-group反复在.a中进行搜索直到所有的未定义字符都被找到为止,而不是默认的只搜索一次。MKL库有这样的问题。

示例

// a.h   
extern void foo();// a.cpp 
#include  
void foo() 
{ printf("foo\n"); 
} // x.cpp 
#include "a.h" 
int main() 
{ foo(); return 0; 
}// Makefile 
all: x 
x: x.cpp liba.so g++ -g -o $@ $^ 
liba.so: liba.a g++ -g -fPIC -shared -o $@ $^ #g++ -g -fPIC -shared -o $@ -Wl,--whole-archive $^ -Wl,-no-whole-archive 
liba.a: a.o ar cru $@ $^ 
a.o: a.cpp g++ -g -c $^ 
clean: rm -f x a.o liba.a liba.so 

运行测试:

$ make g++ -g -c a.cpp ar cru liba.a a.o g++ -g -fPIC -shared -o liba.so liba.a #g++ -g -fPIC -shared -o liba.so -Wl,--whole-archive liba.a -Wl,-no-whole-archive g++ -g -o x x.cpp liba.so /tmp/cc6UYIAF.o: In function `main': /data/jayyi/ld/x.cpp:5: undefined reference to `foo()' collect2: ld returned 1 exit status make: *** [x] Error 1 

  默认情况下,对于未使用到的符号(函数是一种符号),链接器不会将它们链接进共享库和可执行程序。
  这个时候,可以启用链接参数“--whole-archive”来告诉链接器,将后面库中所有符号都链接进来,参数“-no-whole-archive”则是重置,以避免后面库的所有符号被链接进来。
  主要作用就是解决主目录函数与模块间函数相互调用问题。
修改一下makefile:

// Makefile 
all: x 
x: x.cpp liba.so g++ -g -o $@ $^ 
liba.so: liba.a g++ -g -fPIC -shared -o $@ -Wl,--whole-archive $^ -Wl,-no-whole-archive 
liba.a: a.o ar cru $@ $^ 
a.o: a.cpp g++ -g -c $^ 
clean: rm -f x a.o liba.a liba.so 

参考链接

https://www.cnblogs.com/vaughnhuang/p/16474361.html
https://blog.csdn.net/lanhuazui10/article/details/107895036
https://www.cnblogs.com/JimmyTY/p/5856217.html


http://www.ppmy.cn/devtools/103928.html

相关文章

Nosql数据库redis集群配置详解

一、Redis的安装 环境介绍: 一主双从:10(redis-node1)主,20(redis-node2) 30(redis-node3)从——使用的是红帽9.1系统 源码安装redis [rootredis-node1 ~]# tar zxf red…

计算图像分割mask的灰度级个数、以及删除空的分割数据

1、查询mask类别 图像分割的mask是图片格式的话,往往是阈值图像,这样有几个阈值就是分割几个类别的 这里提供代码检测mask的类别个数 摆放如上(只要有mask即可),然后copy下面代码,放在data的同级目录即可…

C#桌面开发之巅:Windows Forms与WPF深度解析

标题:C#桌面开发之巅:Windows Forms与WPF深度解析 摘要 C#作为.NET框架的核心语言之一,提供了强大的桌面应用程序开发能力。Windows Forms和WPF(Windows Presentation Foundation)是两个主要的UI框架,用于…

企业级Mysql 集群技术部署

目录 1.1部署mysql 1.1.1 安装依赖性: 1.1.2 下载并解压源码包 1.1.3 源码编译安装mysql 1.1.4 部署mysql 2.mysql的主从复制 2.1 配置masters 2.2配置slave 2.3 延迟复制 2.4 慢查询日志 2.5并行复制 2.6 原理刨析 2. 7架构缺陷 3.半同步模式 3.1半同…

将Keil工程文件移植到VScode+Platformio环境下

1创建工作空间 1.1创建PIO工程 若弹出报错不用管他(装加密软件的电脑会报错并少创建几个文件)。PIO会在我们选定的文件夹下创建名为Name(Project)的文件夹 打开demo/Project文件夹可以看到生成一堆文件 1.2创建CubeMX工程 网上很…

数据库课程设计mysql

数据库课程设计是一个综合性的过程,它涉及到需求分析、概念设计、逻辑设计、物理设计、实现、测试以及维护等多个阶段。在这里,我将以MySQL为例,详细阐述一个假设的数据库课程设计过程,内容将涵盖从需求分析到实现的具体步骤&…

56基于SpringBoot+Vue+uniapp的教学资源库的详细设计和实现(源码+lw+部署文档+讲解等)

文章目录 前言详细视频演示具体实现截图技术栈后端框架SpringBoot前端框架Vue持久层框架MyBaitsPlus 系统测试系统测试目的系统功能测试系统测试结论 为什么选择我代码参考数据库参考源码获取源码获取 前言 🌞博主介绍 :✌全网粉丝15W,CSDN特邀作者、21…

如何让私域服务赢得用户的心?

私域流量的概念在当今的商业环境中已经变得极为重要,许多品牌和企业都投入大量资源尝试通过各种策略吸引并保留用户。然而,单纯的流量积累并不足以确保商业成功。当面对用户的沉默、缺乏活跃度以及无法变现的困境时,我们必须重新审视私域流量…