37

我制作了一个应用程序(一个可执行文件)成功调用了一些 dylib。但是,dylib 文件和可执行文件位于不同的目录中。我将包含 dylib 文件的目录添加到$PATH环境变量中。但是,它仍然无法加载。我将所有dylib文件复制到可执行文件中,程序终于运行了。这确认 dylib 文件没有问题。但是,我怎样才能告诉操作系统找到它呢?
在 Windows 中,我只需要将包含 dll 文件的目录路径添加到 $PATH。对于 Mac OS X,我需要做什么?

4

4 回答 4

40

在阅读了贾斯汀提供的链接后,我成功地使用@executable_path令牌将我的 dylib install_name 更改为指向我的可执行文件所在的同一个目录。

@executable_path绝对路径很烦人。有时您希望将框架嵌入到应用程序中,而不必将框架安装到 /Library 或类似位置。

Mac 对此的解决方案是@executable_path。这是一个神奇的标记,当放置在库安装名称的开头时,它会扩展为加载它的可执行文件的路径,减去最后一个组件。例如,假设 Bar.app 链接到 Foo.framework。如果 Bar.app 安装在 /Applications 中,@executable_path 将扩展为 /Applications/Bar.app/Contents/MacOS。如果您打算在 Contents/Frameworks 中嵌入框架,则只需将 Foo.framework 的安装名称设置为 @executable_path/../Frameworks/Foo.framework/Versions/A/Foo。动态链接器会将其扩展到 /Applications/Bar.app/Contents/MacOS/../Frameworks/Foo.framework/Versions/A/Foo 并在那里找到框架。

http://www.mikeash.com/pyblog/friday-qa-2009-11-06-linking-and-install-names.html

我将用一个例子来演示。

假设我有以下可执行文件/opt/local/bin/convert,它的 dylib 位于/opt/local/lib中。我想将它复制到另一个目录,并让它从与我复制可执行文件的目录相同的目录中加载它的 dylib。

> mkdir ~/tmp/bin
> cp /opt/local/bin/convert ~/tmp/bin

获取可执行文件 dylibs 的列表

> otool -L ~/tmp/bin/convert
~/tmp/bin/convert:
    /opt/local/lib/libtiff.3.dylib (compatibility version 13.0.0, current version 13.5.0)
    /usr/lib/libSystem.B.dylib (compatibility version 1.0.0, current version 159.1.0)
    /opt/local/lib/libjpeg.8.dylib (compatibility version 12.0.0, current version 12.0.0)
    /opt/local/lib/libfontconfig.1.dylib (compatibility version 6.0.0, current version 6.4.0)
    /opt/local/lib/libiconv.2.dylib (compatibility version 8.0.0, current version 8.1.0)
    /opt/local/lib/libfreetype.6.dylib (compatibility version 15.0.0, current version 15.0.0)
    /opt/local/lib/libexpat.1.dylib (compatibility version 7.0.0, current version 7.2.0)
    /opt/local/lib/libbz2.1.0.dylib (compatibility version 1.0.0, current version 1.0.6)
    /opt/local/lib/libz.1.dylib (compatibility version 1.0.0, current version 1.2.6)
    ...

我只关心/opt/local/lib目录中的 dylib,所以我们只提取/opt中的 dylib 。我想保持所有其他 dylib 引用完整,尤其是对/usr/lib/libSystem的东西。

> DYLIBS=`otool -L ~/tmp/bin/convert | grep "/opt" | awk -F' ' '{ print $1 }'`

将可执行文件引用的所有 dylib 复制到可执行文件已复制到的同一目录中。

> for dylib in $DYLIBS; do cp $dylib ~/tmp/bin/; done;

使用install_name_tool更改我们在上述步骤中提取的所有 dylib 的安装名称,并通过@executable_path在 dylib 名称前面添加 来替换它们。这将使动态链接器在与可执行文件所在的目录相同的目录中查找 dylib。

> for dylib in $DYLIBS; do install_name_tool -change $dylib @executable_path/`basename $dylib` ~/tmp/bin/convert; done;

确认安装名称已更改并且libSystem仍指向/usr/lib/libSystem

> otool -L ~/tmp/bin/convert
~/tmp/bin/convert:
    @executable_path/libtiff.3.dylib (compatibility version 13.0.0, current version 13.5.0)
    /usr/lib/libSystem.B.dylib (compatibility version 1.0.0, current version 159.1.0)
    @executable_path/libjpeg.8.dylib (compatibility version 12.0.0, current version 12.0.0)
    @executable_path/libfontconfig.1.dylib (compatibility version 6.0.0, current version 6.4.0)
    @executable_path/libiconv.2.dylib (compatibility version 8.0.0, current version 8.1.0)
    @executable_path/libfreetype.6.dylib (compatibility version 15.0.0, current version 15.0.0)
    @executable_path/libexpat.1.dylib (compatibility version 7.0.0, current version 7.2.0)
    @executable_path/libbz2.1.0.dylib (compatibility version 1.0.0, current version 1.0.6)
    @executable_path/libz.1.dylib (compatibility version 1.0.0, current version 1.2.6)
    ...
于 2012-07-20T18:46:06.910 回答
15

您需要设置DYLD_LIBRARY_PATH环境变量。

dyld 手册页

      This  is  a  colon  separated  list  of directories that contain libraries. The dynamic linker
      searches these directories before it searches the default locations for libraries.  It  allows
      you to test new versions of existing libraries.

      For  each  library  that  a program uses, the dynamic linker looks for it in each directory in
      DYLD_LIBRARY_PATH in turn. If it still can't find the library,  it  then  searches  DYLD_FALL-
      BACK_FRAMEWORK_PATH and DYLD_FALLBACK_LIBRARY_PATH in turn.
于 2011-01-13T05:29:45.930 回答
6

很多种方法。也许这会有所帮助:

http://www.mikeash.com/pyblog/friday-qa-2009-11-06-linking-and-install-names.html

于 2011-01-13T05:30:30.247 回答
2

如果 dylib 位于库的 INSTALL_NAME 指定的位置,它将 Just Work*。

否则,您可以将 dylib 的位置添加到 DYLD_LIBRARY_PATH。您可能想阅读dyld 文档

*) 完全准确地说,它需要位于 DYLD_ROOT_PATH/INSTALL_NAME,但对 DYLD_ROOT_PATH 大惊小怪的情况很少见。

于 2011-01-13T05:29:33.497 回答