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

时间:2016-11-18 20:53:11

标签: c++ linux linker shared-libraries

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

在Windows上,我可以将此基本库创建为静态库,因此单例的副本将放置在每个插件中。 但是在Linux上我遇到了一个相反的问题as this fellow.

到目前为止,我已尝试过以下内容:

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

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

1 个答案:

答案 0 :(得分:3)

最适合您的解决方案是在Windows上执行相同的操作:将基本库编译为存档(静态)库并将其链接到每个插件中。 (这需要使用-fPIC编译基础库。)

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

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

所以,这就是你需要做的事情:

  1. 使用-fPIC -fvisibility=hidden
  2. 编译基础库
  3. 对于您 要从插件中导出的特定功能,请添加__attribute__((visibility("default")))
  4. 完成此操作后,请运行nm -D new-plugin.so并与nm -D old-plugin.so进行比较。您应该看到旧插件导出所有内容,新插件仅导出 您标记为导出的函数。

    另一种方法是使用linker script来控制符号可见性。