问题描述
首先我会尽量用一句话来定义问题,然后我会详细说明。
问题: 有没有办法在 DLL 中创建的类实例上调用非虚拟函数,因为它隐藏在 DLL 中,因此在编译和链接时不知道实现?
背景: 我有一个系统,我将一段代码(一个模块)编译成一个 DLL,并在调用 GetModule(Name) 时按需将这个 DLL 加载到我的应用程序中。
在 Windows 上,加载是通过 LoadLibrary 完成的
void* DynamicLibraryLoaderWin::LoadDynamicLibrary(const std::wstring& file_path)
{
auto* library_handle = LoadLibraryW(file_path.c_str());
if (!library_handle)
{
const auto error = GetLastError();
assert(false);
}
return library_handle;
}
获得句柄后,我可以通过调用 DLL 中导出的 C 函数来创建模块实例,该函数将返回我想要的模块类的实例。
示例模块:
class ModuleExample : public IModule
{
public:
virtual std::string* GetString() const;
};
现在,如果我想在 ModuleExample 上调用 GetString(),我必须将函数设为虚拟。如果我不这样做,最终会出现链接器错误,因为在我的应用程序中某处有代码执行 GetString() 并且链接器找不到实现 -> 因为隐藏在尚不存在的 DLL 中。如果我将函数设为虚拟,链接器不会抱怨,因为一切都通过 V 表,这是编译器和链接器已知的。
现在我们可以争辩说,使所有函数都虚拟化是解决方案,但虚拟函数的开销很小,这对于性能关键代码来说并非微不足道。因此,我正在寻找一种不使所有功能都虚拟化的解决方案。
另一个小的负面影响是我无法将 ModuleExample 设为 final,因为它尝试对虚函数进行去虚拟化,并且链接器再次抱怨。
也许有趣: 我正在使用 CMake 来设置我的模块和应用程序,并且所有模块都被定义为具有 INTERFACE 属性的库,并且还与 INTERFACE 链接。我听说过 MODULE,但还没有尝试。
解决方法
暂无找到可以解决该程序问题的有效方法,小编努力寻找整理中!
如果你已经找到好的解决方法,欢迎将解决方案带上本链接一起发送给小编。
小编邮箱:dio#foxmail.com (将#修改为@)