如何从 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.libmyCproj.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++ 项目中使用它的主要内容,如果未能解决你的问题,请参考以下文章

如何从 DLL 中检索数组变量? (视觉 C++)

如何安全地将字符串引用从 c# 传递到 c++?

如何从DLL中检索数组变量? (Visual C ++)

如何从 c++ 项目中调用用 c# 创建的 dll 文件? [复制]

C++ 从 dll 调用 FORTRAN 子例程

尝试从 C# 调用 C++ dll 时出现格式不正确的异常