这是.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)的不同技术.它们支持真正的动态链接,这意味着,运行链接器以在运行时解析外部名称.