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

2024-02-14

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

在 Windows 上,我可以将此基础库创建为静态库,因此将单例的副本放置在每个插件中。 然而在 Linux 上我遇到了一些相反的问题作为这个家伙。 https://stackoverflow.com/questions/8623657/multiple-instances-of-singleton-across-shared-libraries-on-linux/8626922?noredirect=1#comment68568958_8626922

到目前为止我已经尝试过以下操作:

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

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


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

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

在 Windows 上,除非您明确DLLEXPORT来自插件的功能,它仍然是内部的。在 Linux 上,default相反,当两个共享库导出相同的符号时,第一个加载的获胜。

因此,您需要执行以下操作:

  1. 编译基础库-fPIC -fvisibility=hidden
  2. 对于您需要的特定功能do想要从插件导出,添加__attribute__((visibility("default"))).

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

另一种选择是使用链接描述文件 https://stackoverflow.com/a/8131634/50617控制符号可见性。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

如何创建共享库特定的单例实例 的相关文章

随机推荐