如何创建共享库特定的单例实例

Dan*_*die 4 c++ linux linker shared-libraries

我有一组应用程序插件,它们都链接到一个公共基础库。基础库定义了一个单例,它维护每个插件中所有对象构造函数的列表。

在 Windows 上,我可以将这个基础库创建为静态库,因此每个插件中都会放置一个单例副本。但是在 Linux 上,我和这个家伙有一些相反的问题

到目前为止,我已经尝试了以下方法:

  • 将基础构建为共享库(根据原作者)
  • 使用 -fPIC 将基础构建为静态
  • 使用 -fPIC 将基础构建为静态,在 CMake 中明确删除 -rdynamic

我真的很想通过将单例定义驻留在每个插件的基础库中以使其拥有自己的实例来保持程序结构与现在相同。我已经尝试将定义移动到每个插件中,但我真的很想避免这种情况。基本上我想重现他认为的错误。然而,他在一个头文件中完全定义了他的单例,这对我来说很有意义,每个插件都有自己的类实例,另一方面,我将单例的定义编译到基本库中。

Emp*_*ian 5

对您来说最好的解决方案是做与在 Windows 上做的相同的事情:将基本库编译为存档(静态)库并将其链接到每个插件中。(这需要使用-fPIC.编译基础库。)

这不起作用的原因:您没有控制从插件导出的功能。

在 Windows 上,除非您明确地DLLEXPORT使用插件中的函数,否则它仍然是内部的。在 Linux 上,默认情况相反,当两个共享库导出相同的符号时,第一个加载的获胜。

所以,这是你需要做的:

  1. 编译基础库 -fPIC -fvisibility=hidden
  2. 对于具体的功能,你想从插件出口,增加__attribute__((visibility("default")))

完成此操作后,运行nm -D new-plugin.so并与nm -D old-plugin.so. 您应该看到旧插件导出所有内容,而新插件导出您标记为导出的功能。

另一种替代方法是使用链接描述文件来控制符号可见性。