如何创建共享库特定的单例实例
How to create shared library specific singleton instances
我有一组应用程序插件,它们都 link 到一个公共基础库。基础库定义了一个单例,它维护了每个插件中所有对象构造函数的列表。
在 windows,我可以将这个基础库创建为静态库,因此在每个插件中放置了一个单例副本。
但是在 Linux 我有一些相反的问题 as this fellow.
到目前为止我已经尝试了以下方法:
- 将基础构建为共享库(根据原作者)
- 使用 -fPIC 将基础构建为静态
- 使用 -fPIC 将基础构建为静态,在 CMake 中显式删除 -rdynamic
我真的很想保持程序结构与现在相同,方法是将单例定义驻留在基础库中,让每个插件都有自己的实例。我已经尝试将定义移动到每个插件中,但我真的很想避免这种情况。
本质上我想重现他认为是错误的东西。然而,他在一个头文件中完全定义了他的单例,这对我来说很有意义,每个插件都会有自己的 class 实例化,另一方面,我将单例的定义编译到基础库中。
最适合您的解决方案是做与 Windows 相同的事情:将基础库编译为存档(静态)库,然后 link 将其放入每个插件中。 (这需要用 -fPIC
编译基础库。)
这不起作用的原因:您没有控制从插件导出的功能。
在 Windows 上,除非您明确 DLLEXPORT
来自插件的函数,否则它仍然是内部的。在 Linux 上,default 是相反的,当两个共享库导出相同的符号时,第一个加载的优先。
所以,这是您需要做的:
- 用
-fPIC -fvisibility=hidden
编译基础库
- 对于您想要从插件导出的特定功能,添加
__attribute__((visibility("default")))
。
完成此操作后,运行 nm -D new-plugin.so
并与 nm -D old-plugin.so
进行比较。您应该看到旧插件导出所有内容,而新插件导出仅您标记为导出的功能。
另一种方法是使用 linker script 来控制符号可见性。
我有一组应用程序插件,它们都 link 到一个公共基础库。基础库定义了一个单例,它维护了每个插件中所有对象构造函数的列表。
在 windows,我可以将这个基础库创建为静态库,因此在每个插件中放置了一个单例副本。 但是在 Linux 我有一些相反的问题 as this fellow.
到目前为止我已经尝试了以下方法:
- 将基础构建为共享库(根据原作者)
- 使用 -fPIC 将基础构建为静态
- 使用 -fPIC 将基础构建为静态,在 CMake 中显式删除 -rdynamic
我真的很想保持程序结构与现在相同,方法是将单例定义驻留在基础库中,让每个插件都有自己的实例。我已经尝试将定义移动到每个插件中,但我真的很想避免这种情况。 本质上我想重现他认为是错误的东西。然而,他在一个头文件中完全定义了他的单例,这对我来说很有意义,每个插件都会有自己的 class 实例化,另一方面,我将单例的定义编译到基础库中。
最适合您的解决方案是做与 Windows 相同的事情:将基础库编译为存档(静态)库,然后 link 将其放入每个插件中。 (这需要用 -fPIC
编译基础库。)
这不起作用的原因:您没有控制从插件导出的功能。
在 Windows 上,除非您明确 DLLEXPORT
来自插件的函数,否则它仍然是内部的。在 Linux 上,default 是相反的,当两个共享库导出相同的符号时,第一个加载的优先。
所以,这是您需要做的:
- 用
-fPIC -fvisibility=hidden
编译基础库
- 对于您想要从插件导出的特定功能,添加
__attribute__((visibility("default")))
。
完成此操作后,运行 nm -D new-plugin.so
并与 nm -D old-plugin.so
进行比较。您应该看到旧插件导出所有内容,而新插件导出仅您标记为导出的功能。
另一种方法是使用 linker script 来控制符号可见性。