linux下动态库soname简介

Linux 支持两种类型的库,每一种库都有各自的优缺点。

静态库包含在编译时静态绑定到一个程序的函数。动态库则不同,它是在加载应用程序时被加载的,而且它与应用程序是在运行时绑定的。

静态库较适宜于较小的应用程序,因为它们只需要最小限度的函数。

而对于需要多个库的应用程序来说,则适合使用共享库,因为它们可以减少应用程序对内存(包括运行时中的磁盘占用和内存占用)的占用。

这是因为多个应用程序可以同时使用一个共享库;因此,每次只需要在内存上复制一个库。要是静态库的话,每一个运行的程序都要有一份库的副本。

GNU/Linux 提供两种处理共享库的方法(每种方法都源于 Sun Solaris)。可以动态地将程序和共享库链接并让 Linux 在执行时加载库(如果它已经在内存中了,则无需再加载)。

另外一种方法是使用一个称为动态加载的过程,这样程序可以有选择地调用库中的函数。

使用动态加载过程,程序可以先加载一个特定的库(已加载则不必),然后调用该库中的某一特定函数。这是构建支持插件的应用程序的一个普遍的方法。

Linux下动态库文件的文件名形如 libxxx.so,其中so是 Shared Object 的缩写,即可以共享的目标文件。

关于动态库的实际使用,可以参考c语言中动态库的使用(启动时链接和运行时链接)
soname介绍
gcc -Wl 生成

-Wl选项告诉编译器将后面的参数传递给链接器。-soname则指定了动态库的soname。

-Wl 表示后面的参数也就是-soname,libhello.so.1直接传给连接器ld进行处理。

如:gcc -fPIC -shared -Wl,soname,libhello.so.1 -o libhello.so.1.2 hello.c

用于使用hello.c生成一个名为libhello.so.1.2的共享库,且该共享库的soname为libhello.so.1。
作用

soname的关键功能是它提供了兼容性的标准,当要升级系统中的一个库时,并且新库的soname和老库的soname一样,用旧库链接生成的程序使用新库依然能正常运行。这个特性使得在Linux下,升级使得共享库的程序和定位错误变得十分容易。

在Linux中,应用程序通过使用soname,来指定所希望库的版本,库作者可以通过保留或改变soname来声明,哪些版本是兼容的,这使得程序员摆脱了共享库版本冲突问题的困扰。
使用

可以通过readelf -d来查看每个动态库的SONAME。

在程序执行期间,程序会查找拥有 soname名字的文件,而不是库的文件名,换句话说,soname是库的区分标志。

目的主要是允许系统中多个版本的库文件共存,习惯上在命名库文件的时候通常与soname相同,

如:libxxxx.so.major.minor,其中,xxxx是库的名字,major是主版本号,minor 是次版本号。

一般对于不影响兼容性的升级,只修改次版本号。如果修改了主版本号,可能就不能兼容以前的版本了。

所以,可以使用这种方式构建动态库:gcc -fPIC -shared -Wl,soname,libhello.so.major -o libhello.so.major.minor hello.c

简单的来说,soname指定了库的名字,而不去管生成的是什么名字的库,在做链接时是将这个soname指定的名字加入执行文件中,而程序运行时也是去加载soname指定的库文件名。补:注意,这里说链接时soname指定的名字加入到执行文件(exe)中,指的是编译生成exe时把这个名字存到了exe内部(‘ldd 可执行文件’可以看到),从而exe运行时找这个名字的文件,但并不是指的exe的编译链接动作所链接的目标文件名,这个目标文件名依然是简单的libxx.so。

所以如果程序连接了新升级的库,只需要将这个新库拷贝到目录下面后,对其以soname做一个符号链接就能调用。

如果库升级了,但是程序依旧使用旧的链接库,那么只需对这个将旧库名字软链接到新升级的库中去即可。

对于一个新安装到系统中的库,有时需要执行ldconfig命令,来自动完成动态库的更新。
总结

Linux系统的这种动态库管理方式值得我们在实际项目的动态库管理中使用,既保证了动态库的升级,又能得到方便地使用。
————————————————
补:在嵌入式中,为了简化,编译时直接以soname作为生成的文件名。so的版本信息在代码中体现。

<think>好的,用户现在问的是SONAME是什么。之前我已经详细解释了共享库的版本控制和符号链接的作用,其中提到了SONAME字段在运行时的重要性。现在需要进一步深入解释SONAME的概念和机制。 首先,我需要确保用户已经理解之前的内容,但可能他们需要更明确的定义。SONAME是一个关键的字段,用于确保动态链接库的兼容性。用户可能想知道为什么需要SONAME,它如何影响编译和运行阶段,以及它在版本控制中的作用。 接下来,我需要解释SONAME的全称是Shared Object Name,它是嵌入在共享库中的一个元数据字段。这个字段告诉动态链接器在运行时应该加载哪个版本的库。比如,当程序编译时链接了libpq.so,但实际运行时根据SONAME查找libpq.so.5,这确保了兼容性。 用户可能还关心如何设置和查看SONAME。可能需要提到使用编译器的选项,如gcc的-Wl,-soname,以及如何用objdump或readelf来查看SONAME。此外,应该举例说明不同版本的SONAME如何影响兼容性,比如主版本号变化意味着不兼容,而次版本号变化则保持兼容。 还要注意用户可能的疑问点,比如SONAME和符号链接的关系,以及为什么需要两个符号链接(libpq.so和libpq.so.5)。需要明确区分编译时和运行时的不同用途,以及SONAME如何在这两个阶段起作用。 最后,总结SONAME的作用,强调它是Linux动态链接库版本控制的核心机制,确保程序在更新库时仍能正确运行,避免兼容性问题。需要保持解释清晰,结构分明,帮助用户逐步理解这个概念。</think>### **SONAME 是什么?** **SONAME**(**Shared Object Name**)是 Linux 动态链接库(`.so` 文件)中嵌入的一个**关键元数据字段**,用于标识库的**二进制兼容性版本**。它是 Linux 动态链接机制的核心,确保程序在编译和运行时能正确找到兼容的库版本。 --- ### **1. SONAME 的作用** - **唯一标识兼容性**: SONAME 是库的“兼容性标签”。若两个库的 SONAME 相同,则它们的二进制接口(ABI)兼容,程序可安全替换使用。 - **指导动态链接器**: 程序运行时,动态链接器(如 `ld.so`)会根据 SONAME 查找并加载对应的库文件,而非直接使用编译时链接的库名。 --- ### **2. SONAME 的生成与查看** #### **(1) 如何生成 SONAME?** 在编译共享库时,通过编译器的 `-Wl,-soname` 参数指定: ```bash gcc -shared -Wl,-soname,libpq.so.5 -o libpq.so.5.12 *.o ``` - `-Wl,-soname,libpq.so.5`:将 SONAME 字段写入库文件。 #### **(2) 如何查看 SONAME?** 使用 `objdump` 或 `readelf` 工具: ```bash objdump -p libpq.so.5.12 | grep SONAME # 或 readelf -d libpq.so.5.12 | grep SONAME ``` 输出示例: ``` SONAME libpq.so.5 # 关键字段 ``` --- ### **3. SONAME 的工作机制** #### **阶段 1:编译时** - 开发者编写程序时,通过 `-lpq` 链接库,编译器找到 `libpq.so`(符号链接指向 `libpq.so.5.12`)。 - 编译器会读取 `libpq.so.5.12` 中的 **SONAME 值(`libpq.so.5`)**,并将此值写入最终的可执行文件。 #### **阶段 2:运行时** - 程序运行时,动态链接器根据可执行文件中记录的 **SONAME(`libpq.so.5`)** 查找对应的库文件。 - 找到 `libpq.so.5`(符号链接指向 `libpq.so.5.12`),并加载实际的库文件。 --- ### **4. 示例:SONAME 与版本升级** 假设库从 `libpq.so.5.12` 升级到 `libpq.so.5.13`: 1. **兼容性更新**(仅修复问题或优化,不破坏接口): - **SONAME 保持不变**:仍为 `libpq.so.5`。 - 更新符号链接 `libpq.so.5 → libpq.so.5.13`,旧程序无需重新编译即可使用新库。 2. **不兼容更新**(修改接口或ABI): - **必须修改 SONAME**(如 `libpq.so.6`),表示与旧版本不兼容。 - 旧程序继续依赖 `libpq.so.5`,新程序使用 `libpq.so.6`。 --- ### **5. 符号链接与 SONAME 的关系** - **`libpq.so`**:编译时使用的符号链接,指向最新版本(如 `libpq.so.5.12`)。开发者无需关心具体版本号。 - **`libpq.so.5`**:运行时符号链接,名称必须与 SONAME 字段完全一致。确保程序运行时加载兼容版本。 --- ### **总结** - **SONAME 的本质**:动态链接库的“兼容性身份证”。 - **核心作用**: - **编译时**:告知编译器写入可执行文件的依赖库版本。 - **运行时**:指导动态链接器加载正确版本的库。 - **规则**: SONAME 相同 → 二进制兼容;SONAME 不同 → 不兼容(需重新编译程序)。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值