众所周知,Linux动态库的默认搜索路径是/lib
和/usr/lib
。动态库被创建后,一般都复制到这两个目录中。当程序执行时需要某动态库,并且该动态库还未加载到内存中,则系统会自动到这两个默认搜索路径中去查找相应的动态库文件,然后加载该文件到内存中,这样程序就可以使用该动态库中的函数,以及该动态库的其它资源了。在Linux 中,动态库的搜索路径除了默认的搜索路径外,还可以通过以下三种方法来指定。
方法一:在配置文件/etc/ld.so.conf
中指定动态库搜索路径。
可以通过编辑配置文件/etc/ld.so.conf来指定动态库的搜索路径,该文件中每行为一个动态库搜索路径。每次编辑完该文件后,都必须运行命令ldconfig使修改后的配置生效。我们通过例1来说明该方法。
例1:
我们通过以下命令用源程序pos_conf.c(见程序1)来创建动态库 libpos.so,详细创建过程请参考文[1]。
# gcc -c pos_conf.c
# gcc -shared -fPCI -o libpos.so pos_conf.o
#
#include <stdio.h>
void pos()
{
printf("/root/test/conf/lib\n");
}
程序1: pos_conf.c
接着通过以下命令编译main.c(见程序2)生成目标程序pos。
# gcc -o pos main.c -L. -lpos
#
void pos();
int main()
{
pos();
return 0;
}
程序2: main.c
然后把库文件移动到目录/root/test/conf/lib中。
# mkdir -p /root/test/conf/lib
# mv libpos.so /root/test/conf/lib
#
最后编辑配置文件/etc/ld.so.conf,在该文件中追加一行"/root/test/conf/lib"。
运行程序pos试试。
# ./pos
./pos: error while loading shared libraries: libpos.so: cannot open shared object file: No such file or directory
#
出错了,系统未找到动态库libpos.so。找找原因,原来在编辑完配置文件/etc/ld.so.conf后,没有运行命令ldconfig,所以刚才的修改还未生效。我们运行ldconfig后再试试。
# ldconfig
# ./pos
/root/test/conf/lib
#
程序pos运行成功,并且打印出正确结果。
方法二:通过环境变量LD_LIBRARY_PATH
指定动态库搜索路径(!)。
通过设定环境变量LD_LIBRARY_PATH也可以指定动态库搜索路径。当通过该环境变量指定多个动态库搜索路径时,路径之间用冒号":"分隔。
不过LD_LIBRARY_PATH的设定作用是全局的,过多的使用可能会影响到其他应用程序的运行,所以多用在调试。(LD_LIBRARY_PATH的缺陷和使用准则,可以参考《Why LD_LIBRARY_PATH is bad》)。通常情况下推荐还是使用gcc的-R或-rpath选项来在编译时就指定库的查找路径,并且该库的路径信息保存在可执行文件中,运行时它会直接到该路径查找库,避免了使用LD_LIBRARY_PATH环境变量查找。
下面通过例2来说明本方法。
例2:
我们通过以下命令用源程序pos_env.c(见程序3)来创建动态库libpos.so。
# gcc -c pos_env.c
# gcc -shared -fPCI -o libpos.so pos_env.o
#
#include <stdio.h>
void pos()
{
printf("/root/test/env/lib\n");
}
程序3: pos_env.c
测试用的可执行文件pos可以使用例1中的得到的目标程序pos,不需要再次编译。因为pos_conf.c中的函数pos和pos_env.c中的函数pos 函数原型一致,且动态库名相同,这就好比修改动态库pos后重新创建该库一样。这也是使用动态库的优点之一。
然后把动态库libpos.so移动到目录/root/test/conf/lib中。
# mkdir -p /root/test/env/lib
# mv libpos.so /root/test/env/lib
#
我们可以使用export来设置该环境变量,在设置该环境变量后所有的命令中,该环境变量都有效。
例如:
# export LD_LIBRARY_PATH=/root/test/env/lib
#
但本文为了举例方便,使用另一种设置环境变量的方法,既在命令前加环境变量设置,该环境变量只对该命令有效,当该命令执行完成后,该环境变量就无效了。如下述命令:
# LD_LIBRARY_PATH=/root/test/env/lib ./pos /root/test/env/lib
#
程序pos运行成功,并且打印的结果是"/root/test/env/lib",正是程序pos_env.c中的函数pos的运行结果。因此程序pos搜索到的动态库是/root/test/env/lib/libpos.so。
方法三:在编译目标代码时指定该程序的动态库搜索路径。
还可以在编译目标代码时指定程序的动态库搜索路径。这是通过gcc 的参数"-Wl,-rpath,
"指定(如例3所示)。当指定多个动态库搜索路径时,路径之间用冒号":"分隔。
例3:
我们通过以下命令用源程序pos.c(见程序4)来创建动态库libpos.so。
# gcc -c pos.c
# gcc -shared -fPCI -o libpos.so pos.o
#
#include <stdio.h>
void pos()
{
printf("./\n");
}
程序4: pos.c
因为我们需要在编译目标代码时指定可执行文件的动态库搜索路径,所以需要用gcc命令重新编译源程序main.c(见程序2)来生成可执行文件pos。
# gcc -o pos main.c -L. -lpos -Wl,-rpath,./
#
再运行程序pos试试。
# ./pos
./
#
程序pos运行成功,输出的结果正是pos.c中的函数pos的运行结果。因此程序pos搜索到的动态库是./libpos.so。
以上介绍了三种指定动态库搜索路径的方法,加上默认的动态库搜索路径/lib和/usr/lib,共五种动态库的搜索路径,那么它们搜索的先后顺序是什么呢?
在 介绍上述三种方法时,分别创建了动态库./libpos.so、 /root/test/env/lib/libpos.so和/root/test/conf/lib/libpos.so。我们再用源程序 pos_lib.c(见程序5)来创建动态库/lib/libpos.so,用源程序pos_usrlib.c(见程序6)来创建动态库 /usr/lib/libpos.so。
#include <stdio.h>
void pos()
{
printf("/lib\n");
}
程序5: pos_lib.c
#include <stdio.h>
void pos()
{
printf("/usr/lib\n");
}
程序6: pos_usrlib.c
这样我们得到五个动态库libpos.so,这些动态库的名字相同,且都包含相同函数原型的公用函数pos。但存储的位置不同和公用函数pos 打印的结果不同。每个动态库中的公用函数pos都输出该动态库所存放的位置。这样我们可以通过执行例3中的可执行文件pos得到的结果不同获知其搜索到了哪个动态库,从而获得第1个动态库搜索顺序,然后删除该动态库,再执行程序pos,获得第2个动态库搜索路径,再删除第2个被搜索到的动态库,如此往复,将可得到Linux搜索动态库的先后顺序。程序pos执行的输出结果和搜索到的动态库的对应关系如表1所示:
程序pos输出结果
使用的动态库
对应的动态库搜索路径指定方式
./ |
./libpos.so |
编译目标代码时指定的动态库搜索路径 |
/root/test/env/lib |
/root/test/env/lib/libpos.so |
环境变量LD_LIBRARY_PATH指定的动态库搜索路径 |
/root/test/conf/lib |
/root/test/conf/lib/libpos.so |
配置文件/etc/ld.so.conf中指定的动态库搜索路径 |
/lib |
/lib/libpos.so |
默认的动态库搜索路径/lib |
/usr/lib |
/usr/lib/libpos.so |
默认的动态库搜索路径/usr/lib |
表1: 程序pos输出结果和动态库的对应关系
创建各个动态库,并放置在相应的目录中。测试环境就准备好了。执行程序pos,并在该命令行中设置环境变量LD_LIBRARY_PATH。
# LD_LIBRARY_PATH=/root/test/env/lib
#./pos
./
#
根据程序pos的输出结果可知,最先搜索的是编译目标代码时指定的动态库搜索路径。然后我们把动态库./libpos.so删除了,再运行上述命令试试。
# rm libpos.so
rm: remove regular file `libpos.so'? y
# LD_LIBRARY_PATH=/root/test/env/lib
#./pos
/root/test/env/lib
#
根据程序pos的输出结果可知,第2个动态库搜索的路径是环境变量LD_LIBRARY_PATH指定的。我们再把/root/test/env/lib/libpos.so删除,运行上述命令。
# rm /root/test/env/lib/libpos.so
rm: remove regular file `/root/test/env/lib/libpos.so'? y
# LD_LIBRARY_PATH=/root/test/env/lib
#./pos
/root/test/conf/lib
#
第3个动态库的搜索路径是配置文件/etc/ld.so.conf指定的路径。删除动态库/root/test/conf/lib/libpos.so后再运行上述命令。
# rm /root/test/conf/lib/libpos.so
rm: remove regular file `/root/test/conf/lib/libpos.so'? y
# LD_LIBRARY_PATH=/root/test/env/lib ./pos /lib
#
第4个动态库的搜索路径是默认搜索路径/lib。我们再删除动态库/lib/libpos.so,运行上述命令。
# rm /lib/libpos.so
rm: remove regular file `/lib/libpos.so'? y
# LD_LIBRARY_PATH=/root/test/env/lib
#./pos
/usr/lib
#
最后的动态库搜索路径是默认搜索路径/usr/lib。
综合以上结果可知,动态库的搜索路径搜索的先后顺序是:
1.编译目标代码时指定的动态库搜索路径;
2.环境变量LD_LIBRARY_PATH指定的动态库搜索路径;
3.配置文件/etc/ld.so.conf中指定的动态库搜索路径;
4.默认的动态库搜索路径/lib;
5.默认的动态库搜索路径/usr/lib。
在上述1、2、3指定动态库搜索路径时,都可指定多个动态库搜索路径,其搜索的先后顺序是按指定路径的先后顺序搜索的。对此本文不再举例说明,有兴趣的读者可以参照本文的方法验证。
分享到:
相关推荐
通过压缩包中的"linux动态库及静态库的创建和使用"文件,你可以找到更具体的实践教程和示例代码,包括如何创建、链接和使用这两种类型的库,以及解决可能出现的问题。这些资料将帮助你深入理解和掌握Linux环境下的...
### Linux静态库与动态库详解 #### 一、引言 在Linux环境下,库是一种重要的软件组件,用于封装一组相关的函数或数据结构,以便于在不同的应用程序之间共享代码。库通常分为两大类:静态库(static libraries)和...
- 在使用动态库的项目中添加库路径和链接选项。 - 正确包含头文件并使用库中的接口。 通过以上步骤,你可以有效地在Linux下利用Qt进行动态库的开发和应用。这对于模块化编程和代码复用至关重要,特别是在大型软件...
在 Linux 系统中,动态库的搜索路径可以通过多种方式指定: * 使用 `-L` 选项指定编译时的库路径,例如: ``` gcc main.c -o main -L ./ -lcac ``` * 使用 `LD_PRELOAD` 环境变量指定运行时的库路径,例如: ``` LD...
【Windows和Linux动态库】 动态链接库(Dynamic Link Library, DLL)技术是程序设计中广泛使用的技巧,旨在减小程序体积,节省存储空间,提高运行效率,并增强程序的可扩展性和模块化管理。Windows和Linux系统都支持...
今天编译完程序运行的时候,报一个错误,说库文件不存在,然后我看了一下,程序的当前路径下就有这个库文件,我以前...众所周知,Linux动态库的默认搜索路径是/lib和/usr/lib。动态库被创建后,一般都复制到这两个目录中。当
### Linux动态库详解 #### 一、概述 Linux 动态库是操作系统中不可或缺的一部分,它们使得多个程序能够共享相同的代码段,进而减小程序体积、提高资源利用率。本文将详细介绍 Linux 动态库的基本概念、如何管理和...
总结,生成Linux动态库是通过C语言编程实现的,涉及的关键操作包括编译源代码为对象文件,使用`-shared`选项创建动态库,安装库到系统路径,并更新链接器缓存。正确使用动态库能够提高软件开发的效率和质量。
- **通过编译时选项** `-R` 或 `-rpath` : 这些选项允许开发者在编译程序时指定运行时的动态库路径,这些信息会直接写入到可执行文件中,使得程序在运行时能直接定位到正确的动态库位置,从而避免了使用`LD_LIBRARY_...
"LINUX动态库.so嵌套.so文件"的主题涉及了如何在一个动态库中嵌入另一个动态库,这种技术通常用于代码重用和模块化设计。下面将详细阐述这一知识点。 首先,我们需要理解动态库的工作原理。在Linux中,`.so`文件...
本篇文章将详细探讨Linux中的库搜索路径,并解释如何配置这些路径。 #### 二、库搜索路径机制 在Linux系统中,当一个程序启动时,系统会根据一系列预设的规则来查找所需的共享库文件。这些规则包括: 1. **硬编码...
在本资料中,我们将深入探讨Linux动态库连接的概念、工作原理以及如何在Qt开发环境中使用动态库。 1. **动态库的类型** - `.so` 文件:这是Linux下动态库的标准格式,全称为"Shared Object"。例如,`libmylib.so` ...
在Linux系统中,开发软件时常会遇到需要创建和使用动态库(.so)与静态库(.a)的情况。动态库允许多个程序共享同一份代码,节省内存资源,而静态库则将库代码直接编译进目标程序,不依赖外部环境。本篇将详细介绍在...
2. 运行阶段:在运行程序时,如果动态链接库不在标准库路径中,需要通过环境变量`LD_LIBRARY_PATH`指定库的位置,或者将其安装到系统默认库目录(如`/usr/lib`或`/usr/local/lib`)。 四、动态链接库的调试 使用`...
### Linux静态库及动态库创建及使用 #### 一、基本概念 ##### 1.1 什么是库 在计算机编程领域,**库**(Library)是一系列预编译的代码集合,这些代码通常实现了某些功能或服务,可供其他程序在运行时调用。库可以...
5. **运行程序**:编译完成后,确保运行环境也已经配置了动态库的搜索路径,通常是通过`LD_LIBRARY_PATH`环境变量或者`/etc/ld.so.conf`配置文件来实现。 6. **错误处理**:在使用过程中,要注意处理可能出现的网络...
6. 环境配置:动态库的查找路径可以通过`LD_LIBRARY_PATH`环境变量进行设置,或者使用`ldconfig`工具更新库搜索路径。 在实际应用中,动态库广泛用于系统开发,提供核心服务和功能,如图形界面、网络通信、数据库...
在Linux系统中,C++动态库(也称为共享对象或.so文件)是程序运行时可以加载和使用的代码库。它们允许多个程序共享同一份代码,从而节省内存并简化软件维护。下面将详细介绍如何创建、使用和管理Linux下的C++动态库...
通过学习和实践“linux_dongtai_lib.zip”中的内容,开发者可以熟练掌握Linux动态库的创建、链接和管理,提升程序开发的效率和灵活性。动态库技术在大型软件项目中尤其重要,因为它允许模块化开发,降低代码冗余,...