2020-09-06 隐式调用 显式调用

摘自https://blog.csdn.net/qq_37059136/article/details/80167388

dll 的 生成方式 .def模式

c++动态函数库的创建与调用(.def)

痕忆丶2018-05-02 16:11:28

8215收藏                                                9                                               

分类专栏:                                      静态库和动态库                                 

版权


动态库的创建有两种方法,之前说过了不使用.def的方法,现在说一下使用.def的方法

环境VS2005

写个简单的动态库,里面有两个函数,能实现加法减法

首先创建一个空项目dll,创建头文件

,再创建源文件

,再创建模块定义文件即.def文件

这样创建一个动态库所需的条件就满足了.

在头文件

中写入如下代码

在源文件

中写入如下代码

在源文件的模块定义文件

中写入如下代码

F7生成动态库

在.def中第一行是数据库的名字,第二行是固定语句EXPORTS,导出代码,下面add跟sub后面可以不加@X(这里就有两种方法导出动态库里的函数,后面会说)

隐式调用

在同一解决方案下创建空WIN32控制台项目,在源文件中创建

,其代码如下

运行结果

这样隐式调用就完成了

注意:隐式调用需要用到动态库里的头文件

和动态库文件夹debug里的

,所以.cpp文件夹中一定要包含,可以拷贝两者到.cpp同一文件夹下,也可以用代码中的方法.

显式调用

在同一解决方案下创建WIN32控制台空项目,在源文件中创建

,代码如下

运行结果

在显式调用中,由于使用的是.def的方法,在获取动态库中函数地址时有两种方法

AddFunc ADD1 = (AddFunc)GetProcAddress(hdll,(char*)(1));这是在.def文件中add后面加@1的使用方法,当然你也可以使用通用获取地址方法 AddFunc ADD1 = (AddFunc)GetProcAddress(hdll,"add");

AddFunc ADD1 = (AddFunc)GetProcAddress(hdll,(char*)(1));这句(char*)(1)中1是add后面@的数字,相当于add在动态库中的地址,GetProcAddress的函数原型里1这个位置数据类型是char*类型.

附.def的一些小知识:

.def 文件中的第一条 LIBRARY 语句不是必须的,但LIBRARY 语句后面的 DLL 的名称必须正确,即与生成的动态链接库的名称必须匹配。此语句将 .def 文件标识为属于 DLL。链接器将此名称放到 DLL 的导入库中。

EXPORTS 语句列出名称,可能的话还会列出 DLL 导出函数的序号值。通过在函数名的后面加上 @ 符和一个数字,给函数分配序号值。当指定序号值时,序号值的范围必须是从 1 到 N,其中 N 是 DLL 导出函数的个数。

LIBRARY BTREE

EXPORTS

Insert @1

Delete @2

Member @3

Min @4

“def   文件中的注释由每个注释行开始处的分号   (;)   指定。注释不能与语句共享一行,但可以在多行语句的规范间出现。



vs2010中使用.def文件导出函数时需要的设置 

在vs2010中使用.def文件导出函数时,要在工程属性-链接器-输入-模块定义文件中加入自定义的.def文件名;可用vs的命令提示查看导出函数 dumpbin -exports dllnames

//dll.h

#pragma once

int __stdcall add(int x,int y);

int __stdcall sub(int x,int y);


// no need .def

//extern "C" _declspec(dllexport) int add(int a, int b);

//extern "C" _declspec(dllexport) int sub(int a, int b);


//main.cpp

#include "dll.h"

int _stdcall add(int x,int y)

{

return x+ y;

}

int _stdcall sub(int x,int y)

{

return x - y;

}

//int  add(int x,int y)

//{

//return x+ y;

//}

//

//int  sub(int x,int y)

//{

//return x - y;

//}



所以隐式调用的意思是,在调用的useDLL的源代码中,可以看到用到的只是LIB和.H,


实际上,运行的时候还是需要用.DLL.


显式调用就很明显说需要dll, 里面没有对LIB的依赖。

LoadLibrary("ConsoleApplication_GenDLL.dll");


我之前都是。。。 没有隐式显式的概念。

编译的时候lib, #include ... 在设置中写 dll. 源代码中没有loadlibrary 这样。这个可能是c和c++的差别。maybe.

你可能感兴趣的:(2020-09-06 隐式调用 显式调用)