这次书接上回,前段时间写了一篇《使用cmake构建C/C++项目和动态库》的文章, 传送门。 但是直接通过cmake编译链接后,会有一个问题,那就是需要的.so文件不能更改目录,一旦.so文件目录变了,整个程序就没法运行了,这肯定是不行的。
原因
后来我查一下一下,linux系统中,程序加载运行需要的.so文件是有顺序的
- 环境变量LD_LIBRARY_PATH指定的路径
- gcc 编译时指定的运行时库路径-rpath
- ldconfig 配置文件ld.so.conf指定的路径
- 系统默认库位置 /lib, /usr/lib
如果没有指定so的位置,gcc会自动把当前so所在的目录作为so的连接目录。知道原因了,问题就好解决了
解决办法
先看一下现在的 CMakeLists.txt文件
1cmake_minimum_required(VERSION 3.13.3)
2project(project1 C)
3
4set(CMAKE_C_STANDARD 99)
5
6add_library(shared SHARED library.h library.c)
7
8set(LIBRARY_OUTPUT_PATH ${PROJECT_SOURCE_DIR}/lib)
9
10add_executable(project1 main.c)
11
12target_link_libraries(project1 shared)
我实验了两种办法,一是把.so 文件放到/lib
或者 /usr/lib
中,这也是在安装很多软件时的做法,当使用包管理器安装软件时,需要的.so文件大多是安装到这两个目录下。在一种就是在编译时指定 rpath
的目录,使用相对目录,这样在复制文件的时候,把.so一起复制就可以了。
先用最简单的办法,把so目录放到系统目录下
现在的目录结构如下,程序依赖的libshared.so 在 lib 目录下,现在把 libshared.so 复制到 /lib 目录下。这里有个要注意的地方,复制完后要执行 ldconfig
命令,重新生成缓存,要不然程序依然找不到对应的.so文件
命令如下
sudo mv lib/libshared.so /lib
sudo ldconfig
这时候在运行 project1 不会报错
编译时指定 rpath目录
设置 rpaht 有两种方式
方式1
1set(CMAKE_SKIP_BUILD_RPATH FALSE)
2set(CMAKE_BUILD_WITH_INSTALL_RPATH TRUE)
3set(CMAKE_INSTALL_RPATH $ORIGIN)
通过修改编译后的 install 路径, 让程序在运行时通过程序的相对目录加载.so文件,其中 $ORIGIN 变量是程序的当前目录
方式2
1set_target_properties(project1 PROPERTIES LINK_FLAGS "-Wl,-rpath,./")
方式2更粗暴,直接设置gcc的编译参数,指定rpaht 是当前目录
修改 CMakeLists.txt文件
1cmake_minimum_required(VERSION 3.13.3)
2project(project1 C)
3
4set(CMAKE_C_STANDARD 99)
5
6add_library(shared SHARED library.h library.c)
7
8set(LIBRARY_OUTPUT_PATH ${PROJECT_SOURCE_DIR}/lib)
9
10#方式1
11set(CMAKE_SKIP_BUILD_RPATH FALSE)
12set(CMAKE_BUILD_WITH_INSTALL_RPATH TRUE)
13set(CMAKE_INSTALL_RPATH $ORIGIN)
14
15add_executable(project1 main.c)
16
17#方式2
18#set_target_properties(project1 PROPERTIES LINK_FLAGS "-Wl,-rpath,./")
19target_link_libraries(project1 shared)
重新生成 MakeFile 文件, 然后编译
编译生成的 libshared.so 还是在 lib目录下,先移动到可执行文件的同级目录下
最终目录如图,现在无论怎么复制文件,只要可执行文件和动态库在一个目录下,都以运行了
总结
解决linux下 动态编译的程序找不到动态库的问题,有多种解决办法,这次用了两种
- 把需要的.so文件放到 /lib 或者 /usr/lib 下, 然后执行
ldconfig
命令 - 通过指定 rpath 来决定加载 .so的目录