.so 库是否需要在所有情况下都出现在 运行 时间

Does a .so library need to be present at run time in all cases

我的问题与 .so 共享库有关。我正在构建一个在一台 ubuntu 机器上使用 cmake 但 运行 在另一台 ubuntu 机器上使用应用程序的项目。

在 CMakeLists.txt 文件中,我有以下几行:

project (clientapp)

add_executable(${PROJECT_NAME} ${SOURCES} ${WAKAAMA_SOURCES} ${SHARED_SOURCES})

LINK_DIRECTORIES(/home/user//mraa-master-built/build/src)
target_link_libraries (clientapp libmraa.so)
target_link_libraries(clientapp m)

这些行将两个库 libmraa.so 和数学库添加到可执行文件中,它在另一台机器上成功运行。

我对共享库的理解是它们必须在编译时和应用程序启动时存在。但是我在另一台机器上没有 libmraa.so 文件,应用程序运行正常。我以为它不起作用。

我的假设正确吗?

一般来说,gcc and clang support lazy linking/binding of symbols, but not for entire libraries。这意味着所有共享对象(即:.so 文件)至少应在应用程序启动时出现。 的一个例外是,如果您将 makefile 修改为不针对这些库 link,并且您通过 dlopen()/dlsym() 手动调用库函数等

这些库中单个符号的绑定可以推迟到需要它们时,或者您可以强制在启动时解析所有符号,using -z lazy or -z now, respectively

奇怪的是,您的应用程序在 libmraa.so 不存在的情况下运行。在没有库的情况下,您的应用程序 运行 最可能的两个原因是:

  • 您的应用程序未使用库中定义的任何符号,因此 linker 在构建时忽略库(尝试 ldd app_name 并查看您的库是否存在于列表中ldd).
  • 提供的库
  • 您的构建脚本中有问题,您正在静态 link 访问库的​​ .a 存档。

编辑:为了响应应用程序如何知道如何找到库,您的linker(在这种情况下为ld)将使用rpath 查找以决定在搜索适当的库时使用哪些目录。您可以通过从命令行执行 LD_DEBUG=libs app_name 之类的操作来了解其工作原理。您还可以通过 LD_LIBRARY_PATH=/some/path app_name.

添加额外的路径

Is my assumption correct?

是的。

应用程序仍然运行的原因有两种可能的解释:

  1. 你错了,libmraa.so在机器的某个地方(虽然可能不在你看的地方),或者
  2. 您的编译器默认为 -Wl,--as-needed,您的二进制文件 事实上 并不依赖于 libmraa.so,尽管它出现在您的 link行。

您可以简单地证实或反驳上述任何猜测。

要确认猜测 2,请执行以下操作:

readelf -d clientapp | grep NEED | grep libmraa
# if there is no output, guess 2 is correct

如果猜测 2 错误,要确认猜测 1,请执行此操作(在没有 libmrra.so 的机器上):

ldd clientapp | grep libmraa.so
# if guess 2 is incorrect, and this command produces no output, then
# your dynamic loader is broken, which is very unlikely.