__declspec(dllexport) 和 __declspec(dllimport)

简介: __declspec(dllexport) 和 __declspec(dllimport)

1.简单介绍

__declspec(dllexport) 和 __declspec(dllimport) 是用来导出和导入 DLL 中的符号的。通常,当你创建一个 DLL 时,你会创建一个包含要导出的函数原型和/或类的头文件,并将 __declspec(dllexport) 添加到头文件中的声明内。为了提高代码的可读性,可以为 __declspec(dllexport) 定义宏,并对要导出的每个符号使用此宏1。

#ifdef 是一个预处理指令,用于检查宏是否已定义。在这种情况下,它可以用来确定代码是在 DLL 中编译还是在使用 DLL 的应用程序中编译。如果是在 DLL 中编译,则使用 __declspec(dllexport) 来导出符号;否则,使用 __declspec(dllimport) 来导入符号。

例如,可以这样写头文件:

#define OS_API_IMPORT __declspec(dllimport)
#define OS_API_EXPORT __declspec(dllexport)
#ifdef BUILD_DLL
#define OS_API OS_API_EXPORT // 如果是生成dll工程,那么导出
#else
#define OS_API OS_API_IMPORT // 如果是生成使用dll的工程,那么导入
#endif
class OS_API A {static int a;};

这行代码定义了一个名为 `A` 的类,该类具有一个名为 `a` 的静态整数成员。`OS_API` 是一个宏,它的值取决于是否定义了 `BUILD_DLL` 宏。如果定义了 `BUILD_DLL` 宏,则 `OS_API` 被定义为 `__declspec(dllexport)`,用于导出符号;否则,`OS_API` 被定义为 `__declspec(dllimport)`,用于导入符号。

因此,在编译 DLL 时,只需定义 `BUILD_DLL` 宏,就可以使用 `OS_API` 宏来导出类 `A`。在使用 DLL 的应用程序中,不需要定义 `BUILD_DLL` 宏,可以使用 `OS_API` 宏来导入类 `A`。

可以把一个DLL比作一个商店,而函数和变量就是商店里的商品。`__declspec(dllexport)`就像是商店的招牌,它告诉顾客(其他程序)这家商店有哪些商品(函数和变量)可以购买(使用)。而`__declspec(dllimport)`就像是顾客的购物清单,它告诉商店(DLL)顾客(当前程序)想要购买哪些商品(函数和变量)。这样,商店就能够提供给顾客想要的商品,而顾客也能够在自己的程序中使用这些商品。

2.例子

下面是一个简单的例子,它演示了如何使用`__declspec(dllexport)`和`__declspec(dllimport)`来导出和导入函数。

首先,我们创建一个DLL,它包含一个名为`Add`的函数,用于计算两个整数的和。我们使用`__declspec(dllexport)`来将该函数从DLL中导出。

// MyLibrary.h
#ifdef MYLIBRARY_EXPORTS
#define MYLIBRARY_API __declspec(dllexport)
#else
#define MYLIBRARY_API __declspec(dllimport)
#endif
MYLIBRARY_API int Add(int a, int b);
// MyLibrary.cpp
#include "MyLibrary.h"
int Add(int a, int b) {
    return a + b;
}

然后,我们创建一个程序来使用这个DLL。我们使用`__declspec(dllimport)`来从DLL中导入`Add`函数。

// main.cpp
#include <iostream>
#include "MyLibrary.h"
int main() {
    int result = Add(1, 2);
    std::cout << "The result is: " << result << std::endl;
    return 0;
}

在这个例子中,我们使用了条件编译来定义一个宏`MYLIBRARY_API`,它根据是否定义了`MYLIBRARY_EXPORTS`来决定使用`__declspec(dllexport)`还是`__declspec(dllimport)`。当我们编译DLL时,我们需要定义`MYLIBRARY_EXPORTS`,这样就会使用`__declspec(dllexport)`来导出函数。当我们编译使用DLL的程序时,我们不需要定义`MYLIBRARY_EXPORTS`,这样就会使用`__declspec(dllimport)`来导入函数。

简单讲,就是一个是给编译dll时自身用的,一个是其他程序需要调用dll用的

目录
打赏
0
0
0
0
1
分享
相关文章
推荐场景GPU优化的探索与实践:CUDA Graph与多流并行的比较与分析
RTP 系统(即 Rank Service),是一个面向搜索和推荐的 ranking 需求,支持多种模型的在线 inference 服务,是阿里智能引擎团队沉淀多年的技术产品。今年,团队在推荐场景的GPU性能优化上又做了新尝试——在RTP上集成了Multi Stream,改变了TensorFlow的单流机制,让多流的执行并行,作为增加GPU并行度的另一种选择。本文详细介绍与比较了CUDA Graph与多流并行这两个方案,以及团队的实践成果与心得。
C/C++ 常见函数调用约定(__stdcall,__cdecl,__fastcall等):介绍常见函数调用约定的基本概念、用途和作用
C/C++ 常见函数调用约定(__stdcall,__cdecl,__fastcall等):介绍常见函数调用约定的基本概念、用途和作用
530 0
QCAD v3.23.0.2源码编译,使用VS2017+Qt5.12.5环境
QCAD v3.23.0.2源码编译,使用VS2017+Qt5.12.5环境
1172 0
【实战指南】4步实现C++插件化编程,轻松实现功能定制与扩展
本文介绍了如何通过四步实现C++插件化编程,实现功能定制与扩展。主要内容包括引言、概述、需求分析、设计方案、详细设计、验证和总结。通过动态加载功能模块,实现软件的高度灵活性和可扩展性,支持快速定制和市场变化响应。具体步骤涉及配置文件构建、模块编译、动态库入口实现和主程序加载。验证部分展示了模块加载成功的日志和配置信息。总结中强调了插件化编程的优势及其在多个方面的应用。
977 132
深入理解 CMake 的 `cmake --build` 命令
深入理解 CMake 的 `cmake --build` 命令
1569 1
【C++ 17 包裹类 泛型容器 std::any】深入理解与应用C++ std::any:从泛型编程到多态设计
【C++ 17 包裹类 泛型容器 std::any】深入理解与应用C++ std::any:从泛型编程到多态设计
1189 1
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等

登录插画

登录以查看您的控制台资源

管理云资源
状态一览
快捷访问