如何从 C 代码正确创建 DLL 并在 C++ 项目中使用它
Posted
技术标签:
【中文标题】如何从 C 代码正确创建 DLL 并在 C++ 项目中使用它【英文标题】:How to properly create a DLL from C code and use it in a C++ project 【发布时间】:2017-12-13 00:48:40 【问题描述】:我有一组用 C 编写的函数,我需要能够从另一个用 C++ 编写的项目调用这些函数。 C 代码本质上是一些对大型数据集进行一些计算的函数。它们不是我写的——我想做的只是让我的 C++ 项目能够调用这些函数。我的解决方案是为 C 代码创建一个 DLL 并将其链接到我的 C++ 项目。
为了制作 DLL,我将 myCproj.h(C 项目中的头文件,而不是 C++ 项目中的头文件)构造如下:
#ifdef __cplusplus
extern "C"
#endif
struct __declspec(dllexport) neededStruct
int a;
//I need to be able to initialize this struct in my C++ project.
__declspec(dllexport) void neededFunc( struct neededStruct *input )
//I need to be able to call this function from my C++ project and feed
//it my local instance of neededStruct.
#ifdef __cplusplus
#endif
src 文件 myCproj.c 根本没有更改。函数定义前面没有__declspec(dllexport)
,也没有在任何地方插入extern "C"
。代码编译没有错误并生成 myCproj.dll 和 myCproj.lib。
然后我告诉我在 VS 中的 C++ 项目在哪里可以找到相应的 myCproj.lib 和 myCproj.h,并将 DLL 复制到我的 C++ 可执行文件所在的目录。为了使用 DLL,我给 myCPPproj.cpp 添加了以下内容:
#define DLLImport __declspec(dllimport)
struct DLLImport neededStruct input;
input.a = 0;
extern "C" DLLImport void neededFunc( &input );
但是,我在最后一行收到错误 EO335 'linkage specification is not allowed'。我做错了什么?
【问题讨论】:
myCproj.h
使用__declspec(dllexport)
。这仅在您构建您的库时是正确的。当您使用它时,这是不正确的。您需要有条件地将 __declspec
属性定义为 dllimport 或 dllexport,具体取决于您正在构建的内容。
@n.m 但我正在构建我的库...这是 C 项目的头文件,而不是 C++ 项目。
这有点令人困惑。在您的问题中,您说:“然后我告诉我在 VS 中的 C++ 项目在哪里可以找到 myCproj.lib
和 myCproj.h
”,这意味着您在 C++ 应用程序中包含了 myCproj.h
。但你是说你不这样做?从您显示的代码来看,您确实在 C++ 中的 extern
语句中使用了 __declspec(dllexport)
。也不是这样?
顺便说一句,编写一个 DLL 函数来初始化该结构比直接从 C++ 应用程序访问它更有意义。
@lurker 所以我认为为了使用 DLL,最好使用实际的头文件来制作该 DLL 并访问需要在项目中包含该头文件的库。这叫做隐式链接对吗?我的理解是,仅链接 DLL 会显式链接它,但由于我同时拥有 .lib 和 .h 文件,我不妨隐式链接它。
【参考方案1】:
最好为库和使用代码使用相同的标头。 如前所述,它通常由条件定义完成,如下所示: MyLibrary.h:
#if defined(MYLIBRARY_API)
#define MYLIBRARY_EXPORTS __declspec(dllexport)
#else
#define MYLIBRARY_EXPORTS __declspec(dllimport)
#endif
#if defined(__cplusplus)
extern "C"
#endif
MYLIBRARY_API bool MyLibFunc();
#if defined(__cplusplus)
#endif
MyLibrary.c:
#include "MyLibrary.h"
void MyLibFunc()
....
App.cpp:
#include <MyLibrary.h>
int main()
MyLibFunc();
将为库项目定义符号 MYLIBRARY_API(通常作为编译器命令行上的 /D)。而且,如果您使用 Visual Studio,那么在创建带有导出的 dll 项目时所得到的结果几乎完全一样。
【讨论】:
MYLIBRARY_API
在哪里/如何定义?此外,从命名的角度来看,在某些情况下将MYLIBRARY_EXPORT
定义为__declspec(dllimport)
有点奇怪。我可能会找到一个不同的合适的名字。
有道理,谢谢!但是符号“MYLIBRARY_API”的命名约定是什么?它必须是所有大写的标题名称,并附加“_API”吗?我见过其他人也使用“_EXPORTS”作为后缀。以上是关于如何从 C 代码正确创建 DLL 并在 C++ 项目中使用它的主要内容,如果未能解决你的问题,请参考以下文章