在 DLL 中导出静态数据

新手上路,请多包涵

我有一个 DLL,其中包含一个带有静态 成员 的类。我使用 __declspec(dllexport) 为了利用这个类的 方法。但是当我将它链接到另一个项目并尝试编译它时,我得到静态数据的“未解析的外部符号”错误。

例如在 DLL 中,Test.h

 class __declspec(dllexport) Test{
protected:
    static int d;
public:
    static void m(){int x = a;}
}

在 DLL 中,Test.cpp

 #include "Test.h"

int Test::d;

在使用 Test 的应用程序中,我调用 m()。

我还尝试对每种方法分别使用 __declspec(dllexport) ,但静态成员仍然出现相同的链接错误。

如果我使用 dumpbin 检查 DLL(.lib),我可以看到符号已被导出。

例如,应用程序在链接时给出以下错误:

 1>Main.obj : error LNK2001: unresolved external symbol "protected: static int CalcEngine::i_MatrixRow" (?i_MatrixRow@CalcEngine@@1HA)

但是 .lib 的垃圾箱包含:

 Version      : 0
  Machine      : 14C (x86)
  TimeDateStamp: 4BA3611A Fri Mar 19 17:03:46 2010
  SizeOfData   : 0000002C
  DLL name     : CalcEngine.dll
  Symbol name  : ?i_MatrixRow@CalcEngine@@1HA (protected: static int CalcEngine::i_MatrixRow)
  Type         : data
  Name type    : name
  Hint         : 31
  Name         : ?i_MatrixRow@CalcEngine@@1HA

我不知道如何解决这个问题。我究竟做错了什么?我怎样才能克服这些错误?

PS 该代码最初是为 Linux 开发的,.so/binary 组合可以正常工作

编辑:在给定的情况下,应用程序不直接引用静态变量,但该方法是内联的,因为它在标题中。我能够通过将方法移动到 .cpp 文件来解决链接错误。

原文由 Gayan 发布,翻译遵循 CC BY-SA 4.0 许可协议

阅读 1.5k
2 个回答

在 cprogramming.com 的 这个 线程中,建议静态变量是 dll 的本地变量,而不是导出的。

以下讨论摘要

调用应用程序中的代码不能直接访问静态成员,只能通过dll中类的成员函数访问。然而,有几个 内联 函数访问静态成员。这些函数将被内联扩展到调用应用程序代码中,使调用应用程序直接访问静态成员。这将违反上面提到的发现,即静态变量是 dll 的本地变量,不能从调用应用程序中引用。

原文由 Anders Abel 发布,翻译遵循 CC BY-SA 4.0 许可协议

我的猜测是使用 DLL 的类应该在标题中看到 dllimport 而不是 dllexport 。如果我是正确的,这通常可以通过定义一个预处理器宏来实现,例如:

 #ifdef EXPORTING
#define DECLSPEC __declspec(dllexport)
#else
#define DECLSPEC __declspec(dllimport)
#endif

然后在类声明中使用它:

 class DECLSPEC Test{
protected:
    static int d;
public:
    static void m(){}
}

因此,在 Test.cpp(或在您的 DLL 项目中有意义的任何地方)中,您可以指定要导出,以便使用 dllexport 导出:

 #define EXPORTING
#include "Test.h"

int Test::d;

而另一个未定义 EXPORTING 的项目将看到 dllimport

是否有意义?

原文由 Ghislain Fourny 发布,翻译遵循 CC BY-SA 2.5 许可协议

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题