.dylib是macOS上的动态库扩展名,但我从来都不清楚什么时候不能/不应该使用传统的unix .so共享对象。
我有一些问题:
在概念层面上,.so和.dylib之间的主要区别是什么? 什么时候我可以/应该使用其中一种而不是另一种? 编译技巧和提示(例如,gcc -shared -fPIC的替换,因为它在osx上不起作用)
.dylib是macOS上的动态库扩展名,但我从来都不清楚什么时候不能/不应该使用传统的unix .so共享对象。
我有一些问题:
在概念层面上,.so和.dylib之间的主要区别是什么? 什么时候我可以/应该使用其中一种而不是另一种? 编译技巧和提示(例如,gcc -shared -fPIC的替换,因为它在osx上不起作用)
Mac OS X用于可执行文件和库的Mach-O对象文件格式区分了共享库和动态加载模块。使用otool -hv some_file查看some_file的文件类型。
Mach-O共享库的文件类型为MH_DYLIB,扩展名为.dylib。它们可以被链接到通常的静态链接器标志,例如libfoo.dylib的-lfoo。它们可以通过将-dynamiclib标志传递给编译器来创建。(-fPIC是默认值,不需要指定。)
可加载模块在Mach-O语言中称为“包”。它们的文件类型为MH_BUNDLE。它们可以携带任何扩展;苹果推荐扩展名.bundle,但大多数移植软件为了兼容性使用.so。通常,您将为扩展应用程序的插件使用包;在这种情况下,bundle将链接到应用程序二进制文件,以访问应用程序导出的API。它们可以通过将-bundle标志传递给编译器来创建。
dylibs和bundle都可以使用dl api动态加载(例如dlopen, dlclose)。不可能像链接共享库一样链接包。然而,也有可能一个bundle链接到真正的共享库;这些将在bundle加载时自动加载。
Historically, the differences were more significant. In Mac OS X 10.0, there was no way to dynamically load libraries. A set of dyld APIs (e.g. NSCreateObjectFileImageFromFile, NSLinkModule) were introduced with 10.1 to load and unload bundles, but they didn't work for dylibs. A dlopen compatibility library that worked with bundles was added in 10.3; in 10.4, dlopen was rewritten to be a native part of dyld and added support for loading (but not unloading) dylibs. Finally, 10.5 added support for using dlclose with dylibs and deprecated the dyld APIs.
在像Linux这样的ELF系统上,两者都使用相同的文件格式;任何一段共享代码都可以用作库和动态加载。
最后,要注意,在Mac OS X中,“bundle”也可以指具有标准化结构的目录,其中包含可执行代码和该代码所使用的资源。有一些概念上的重叠(特别是与“可加载包”,如插件,通常包含Mach-O包形式的可执行代码),但它们不应该与上面讨论的Mach-O包混淆。
额外的引用:
Fink移植指南,这个答案的基础(虽然相当过时,因为它是为Mac OS X 10.3编写的)。 Ld(1)和dlopen(3) 动态库编程主题 Mach-O编程主题
.so文件不是共享库的UNIX文件扩展名。
它只是碰巧是一个常见的问题。
检查ArnaudRecipes sharedlib页面的3b行
基本上,.dylib是用来指示共享库的mac文件扩展名。
在mac os x上,.dylib和.so的区别在于它们是如何编译的。对于.so文件使用-shared,对于.dylib文件使用-dynamiclib。.so和. DYLIB都可以作为动态库文件互换,它们的类型都是DYLIB或BUNDLE。下面是不同文件的读数显示这一点。
libtriangle.dylib:
Mach header
magic cputype cpusubtype caps filetype ncmds sizeofcmds flags
MH_MAGIC_64 X86_64 ALL 0x00 DYLIB 17 1368 NOUNDEFS DYLDLINK TWOLEVEL NO_REEXPORTED_DYLIBS
libtriangle.so:
Mach header
magic cputype cpusubtype caps filetype ncmds sizeofcmds flags
MH_MAGIC_64 X86_64 ALL 0x00 DYLIB 17 1256 NOUNDEFS DYLDLINK TWOLEVEL NO_REEXPORTED_DYLIBS
triangle.so:
Mach header
magic cputype cpusubtype caps filetype ncmds sizeofcmds flags
MH_MAGIC_64 X86_64 ALL 0x00 BUNDLE 16 1696 NOUNDEFS DYLDLINK TWOLEVEL
这两个在Mac OS X上是等价的原因是为了向后兼容其他编译为.so文件类型的UNIX操作系统程序。
编译注意事项:无论编译的是.so文件还是.dylib文件,都需要在链接步骤中将正确的路径插入到动态库中。可以通过向链接命令添加-install_name和文件路径来实现这一点。如果你不这样做,你会遇到在这篇文章中看到的问题:Mac动态库的疯狂(可能是Fortran Only)。
我在OSX上用cmake构建简单代码时观察到:
cmake ... -DBUILD_SHARED_LIBS=OFF ...
创建.so文件
而
cmake ... -DBUILD_SHARED_LIBS=ON ...
创建.dynlib文件。
也许这对任何人都有帮助。