【发布时间】:2018-03-30 21:38:29
【问题描述】:
我在 Dll 中包装了一个静态库来隐藏很多实现的东西,因为只需要 4-5 个函数并避免提供所有第三方库和许多头文件。我似乎在将函数从静态库导出到 dll 时遇到问题。 静态库具有类似于下面的设置类/结构
struct FooSettings
{
bool Read(const std::string& file); // implemented in .cpp
bool Write(const std::string& file); // implemented in .cpp
// rest members, just plain types
};
在Dll端
#include "FooSettings.h"
#if defined(WIN32) || defined(_WIN32)
#if defined(LIB_EXPORT)
// DLL Build, exporting symbols
#define LIB_API __declspec(dllexport)
#elif LIB_IMPORT
// DLL use, importing symbols
#define LIB_API __declspec(dllimport)
#endif
#endif
#ifndef LIB_API
#define LIB_API
#endif
class LIB_API LibSDK
{
public:
LibSDK();
~LibSDK();
FooSettings get() const noexcept;
void set(const FooSettings& settings) const noexcept;
void dummy()
{
foo.Read("");
}
private:
// etc...
};
我可以在“客户端”端拨打dummy(),没有任何问题
但下面的代码会导致无法解析的符号
FooSettings foo;
foo.Read("");
我原以为 FooSettings:Read 至少会被导出,因为它是虚拟函数的一部分。我错过了什么吗?我的偏好是在没有虚拟函数的情况下导出它,但我似乎无法让它工作。
【问题讨论】:
-
如何导出函数?为什么你的代码中没有
__declspec(dllexport)?你在使用导出文件吗?您应该发布完整的代码。此外,如果您需要导出Read函数,您可以在不导出dummy的情况下执行此操作。两者之间没有任何关系。 -
@Ivan 我已经编辑了我的帖子以展示我如何导出类,正如你所说,它几乎是通过定义 LIB_EXPORT 使用 __declspec(dllexport)。问题出在静态库中的类函数
-
所以你正在导出一个
LibSDK类(tihs 包含dummy函数,但你没有导出FooSettings。你希望FooSettings自动导出吗?或者你想要导出LibSDK而不导出FooSettings?在这种情况下dummy不应该是inline函数。 -
@Ivan,FooSettings 函数是静态库的一部分,因此没有必要导出它。据我所知,来自blogs.msdn.microsoft.com/oldnewthing/20140321-00/?p=1433,您要么需要一个 .def 文件,要么需要创建一个虚拟函数。所以在这种情况下,我正在尝试一个虚拟函数。我假设由于虚拟函数可以看到调用读取和写入的符号,那么我应该能够从“客户端”端这样做,但这会导致符号无法解析。当然,虚拟函数在 .cpp 文件中,但在这里为了简洁起见,我将它们全部添加在一起。
-
我应该说是的,我知道如果我在 Dll 项目中添加所有静态库文件并导出 FooSettings 类,那么一切都可以正常工作。我只是想知道是否可以避免这样做并强制以不同的方式添加符号。我还读到您可以使用链接器属性中的 \Includes 来做到这一点,但我无法弄清楚语法。