我有一个很大的项目,在运行时,生成可执行的A.exe
(带有CMAKE工具链(,以动态加载B.dll
。但是如果没有与B.dll
链接的编译时,我在构建可执行文件时得到了"未解决的外部符号"。
这样组织的项目:
superproject
|
|--- build/
|----CMakeLists.txt
|--- src/
|
|---a : a1.cpp, CMakeLists.txt
|---b : b.h, b.cpp, export.def, CMakeLists.txt
文件 src/b/b.h :
class Machine
{
int u;
static Machine* sing;
public:
Machine();
static Machine* GetInstance();
int answer();
};
文件 src/b/b.cpp :
#include "b.h"
Machine* Machine::sing;
Machine::Machine() : u(1) {}
Machine* Machine::GetInstance()
{
if (!sing)
sing = new Machine();
return sing;
}
int Machine::answer(){ return u + 33; }
Machine* GetMach()
{
return Machine::GetInstance();
}
文件 src/a/a1.cpp :
#include <windows.h>
#include <iostream>
#include "b.h"
using namespace std;
typedef Machine* (*func_t)();
int main(int argc, char* argv[])
{
cout << "hellon";
HMODULE lib = LoadLibrary("b.dll");
if (lib)
{
void* rawfunc = GetProcAddress(lib, "GetMach");
if (rawfunc)
{
func_t func = (func_t)(rawfunc);
Machine* m = func();
cout << "The exported number is " << m->answer();
}
else
{
cout << "Failed to get the symbol";
}
}
else
{
cout << "Failed to load library";
}
getchar();
return 0;
}
文件 src/b/export.def :
EXPORTS
GetMach
文件 src/b/cmakelists.txt :
file(GLOB_RECURSE SRC_FILES *.cpp)
file(GLOB_RECURSE DEF_FILES *.def)
add_library(b MODULE ${SRC_FILES} ${DEF_FILES})
文件 src/a/cmakelists.txt :
file(GLOB_RECURSE SRC_FILES *)
include_directories(../b)
add_executable(A ${SRC_FILES})
Visual Studio抱怨:
error LNK2019: unresolved external symbol "public: int __cdecl Machine::answer(void)" (?answer@Machine@@QEAAHXZ) referenced in function main C:....superprojbuildaa1.obj A
为什么我会遇到此错误以及如何消除它?
从链接器的角度来看, non-virtual 方法(Machine::answer
(与全局函数相同(例如,GetMach()
(。
也就是说,由于能够使用此方法,A
应该:
- 链接(
target_link_libraries
(带有定义方法或
的库 - 请求该方法在运行时(
GetProcAddress()
(。
如果这两种方法似乎都不一致,则可以将该方法声明为 virtual :
virtual int answer();
所有虚拟方法是在运行时上解析的。链接器不必费心使用虚拟方法。