运行时动态加载库时出错"unresolved external symbol"



我有一个很大的项目,在运行时,生成可执行的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();

所有虚拟方法是在运行时上解析的。链接器不必费心使用虚拟方法。

相关内容

最新更新