题
我读过很多关于 C++ 中非托管 DLL 的教程/文章。然而,在我的一生中,我似乎无法理解这个概念。我很容易对它是否需要头文件、如何导出它、我是否需要 .lib 文件以及你有什么等看似分歧感到困惑。
所以,假设我只有一个像这样的函数:
public int calculateSquare(int num)
{
return num*num;
}
忽略实际的代码,我需要什么才能将这个简单的函数本身变成一个可供我调用的 DLL?我是否只需将 __dllexport 或其他内容添加到第一行,或者是否需要标头?我对这一切感到困惑。
解决方案
我不能强调这一点,C ++编译器没有看到头文件,在预处理器完成后,只有一个大的源文件(也称为编译单元)。因此严格来说,您不需要标头来从dll导出此函数。 您需要的是某种形式的条件编译,以便在您正在编译的dll中导出函数并将其导入客户端代码中。
通常,这是通过宏和头文件的组合来完成的。您创建一个名为MYIMPORTEXPORT的宏,并通过使用宏条件语句使其在dll中与__declspec(dllexport)一样工作,在客户端代码中使用__declspec(dllimport)。
在文件MYIMPORTEXPORT.h中
#ifdef SOME_CONDITION
#define MYIMPORTEXPORT __declspec( dllexport )
#else
#define MYIMPORTEXPORT __declspec( dllimport )
#endif
文件MyHeader.h中的
#include <MyImportExport.h>
MYIMPORTEXPORT public int calculateSquare(int num)
{
return num*num;
}
在dll .cpp文件中
#define SOME_CONDITION
#include <MyHeader.h>
客户端代码.cpp文件中的
#include <MyHeader.h>
当然,您还需要向链接器发出信号,表明您正在使用 / DLL选项。
构建过程也将生成一个.lib文件,这是一个静态库 - 在这种情况下称为存根 - 客户端代码需要链接到它,就像链接到真正的静态库一样。自动地,将在运行客户端代码时加载dll。当然,操作系统需要通过其查找机制找到dll,这意味着你不能将dll放在任何地方,而是放在特定的位置。 此处更多内容。
一个非常方便的工具,用于查看是否从dll导出了正确的函数,以及客户端代码是否正确导入 dumpbin 。分别用/ EXPORTS和/ IMPORTS运行它。
其他提示
QBziZ的答案是对的。请参阅 C ++中的非托管DLL
要完成它:在C ++中,如果需要使用符号,则必须告诉编译器它存在,并且通常是它的原型。
在其他语言中,编译器只会自己探索库,并找到符号 et voil&#224; 。
在C ++中,您必须告诉编译器。
将C / C ++标题视为书目目录
最好的方法是在一些常见的地方放置所需的代码。如果你愿意,可以使用“界面”。这通常在头文件中完成,称为头文件,因为这通常不是一个独立的源文件。标题只是一个文件,其目的是包含(即由预处理器复制/粘贴)到真正的源文件中。
实质上,你似乎必须声明两次符号(函数,类,等等)。与其他语言相比,这几乎是一种异端。
您应该将其视为一本书,包含摘要表或索引。在表格中,您有所有章节。在文中,您有章节及其内容。
有时,你很高兴你有章节清单。
在C ++中,这是标题。
DLL怎么样?
所以,回到DLL问题:DLL的目的是导出代码将使用的符号。
因此,以C ++方式,您必须在编译时导出代码(例如,在Windows中,使用__declspec,例如)和“发布”。导出内容的表(即具有包含导出声明的“public”标题)。
导出函数清单:
- 调用约定是否适合调用者?(这决定了参数和结果如何传递,以及谁负责清理堆栈)。您应该明确说明您的调用约定。
- 符号将以哪个名称导出?C++ 通常需要修饰(“mangle”)符号的名称,例如来区分不同的重载。
- 告诉链接器使该函数作为 DLL 导出可见
在 MSVC 上:
__stdcall
(这是 pascal 调用约定)是导出符号的典型调用约定 - 我猜大多数客户端都支持。- extern "C" 允许您导出 C 风格的符号而无需名称修改
- 使用
__declspec(dllexport)
标记要导出的符号,或链接一个单独的 .def 文件,其中列出了要导出的符号。使用 .def 文件,您还可以仅按序数(而不是按名称)导出,并更改导出的符号的名称。
您需要使用 __ declspec(dllexport)
导出函数或将函数添加到模块定义文件(.def)。然后将该项目编译为DLL。
在客户端,您有两种选择。使用编译DLL时生成的导入库(.lib)。只需使用此库链接您的客户端项目,即可访问从DLL导出的函数。而且你需要一个头文件,因为编译器需要知道你的函数的签名 - 它返回int并获取一个int。回顾一下,您需要链接一个导入库(.lib)和一个包含函数头的头文件。
另一种方法是使用 WinAPI
调用 LoadLibrary
然后 GetProcAddress
动态加载DLL以获取指向该函数的指针。指向函数的指针必须具有正确的类型,以便编译器可以为其提供正确的参数,并使用正确的调用约定。