摘自https://blog.csdn.net/qq_37059136/article/details/80167388
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.