c – 未解析的外部符号静态变量(标头中定义的方法使用的变量)

这是.h:

class Logger
{
private:
    static int mTresholdSeverity;

public:
    static __declspec(dllexport) void log(const char* message);
    static __declspec(dllexport) void logFormat(const char* format, ...);

    static __declspec(dllexport) int getTresholdSeverity() { return mTresholdSeverity; }
    static __declspec(dllexport) void setTresholdSeverity(int tresholdSeverity) { mTresholdSeverity = tresholdSeverity; }
};

而.cpp:

#include "Logger.h"
#include <cstdarg>

int Logger::mTresholdSeverity = 200;

void Logger::log(const char* message)
{
    //...
}

void Logger::logFormat(const char* format, ...)
{
    //...
}

我收到此错误:
错误LNK2001:未解析的外部符号“private:static int TransformationViewer_Utility_Logging :: Logger :: mTresholdSeverity”(?mTresholdSeverity @ Logger @ TransformationViewer_Utility_Logging @@ 0HA)…

显然,mTresholdSeverity已初始化.如果我注释掉getTresholdSeverity()和setTresholdSeverity(),或者我将它们的定义移动到.cpp文件中,则会删除该错误.

当头文件(getTresholdSeverity()或setTresholdSeverity())中定义的静态方法使用静态变量(mTresholdSeverity)时,为什么会出现链接错误?

最佳答案 下面是它的工作原理.

每个DLL(或EXE)或其他完整的“完全链接”二进制文件必须具有所有引用名称的定义,包括静态变量,包括C类中的静态数据成员.

它们将在应用程序中的DLL之间分开.这意味着,此变量值将有所不同,具体取决于您查找的DLL.将函数移动到CPP文件将使它们执行不同的操作:它们现在将看到DLL的变量副本而不是EXE的副本.

为了使你编写的内容编译,必须在一个地方的所有用户二进制文件中存在CPP的定义.这意味着,DLL和DLL的用户(EXE或DLL)必须具有一个具有此定义的CPP文件.

这是一个非常大的麻烦,因为它使Singleton模式不可能(为程序中的所有用户提供共享数据对象),并且DLL的每个副本必须具有其自己的静态.这种问题仅存在于Windows上,其中DLL是动态*加载的*库.在UNIX系统上,有一种称为共享对象(SO)的不同技术.它们支持真正的动态链接,这意味着,运行链接器以在运行时解析外部名称.

点赞