个人建议:能使用静态库的就不要使用动态库,能使用隐式调用的就不要用显示调用。
注意:
(1)动态库中的.lib文件叫做导入库,对于导入库而言,其实际的执行代码位于动态库中,导入库只包含了地址符号表等,确保程序找到对应函数的一些基本地址信息。
静态库中的.lib叫做静态库,本身就包含了实际执行代码、符号表等等
(2)显示调用一定要用extern “C” 变为C标准编译,可杜绝C++的重载
静态库:在链接阶段将汇编生成的目标文件.o与引用库一起链接打包到可执行文件中,可简单看成(.o或者.obj文件的集合)。(1)对函数库的链接是放在编译时期完成的(2)程序在运行时与函数库没有瓜葛,移植方便(3)浪费空间和资源
静态库的创建:创建Win32项目-》Lib-》添加C++类-》添加函数
static double add(double a,double b);
调用:(1)若为第三方库-》VC++目录-》包含目录,库目录
链接器-》输入-》附加依赖项
(2)若在同一个工程中-》通用属性-》框架和引用-》添加新引用
动态库:
(1)将库函数的链接载入推迟到程序运行时期
(2)可以实现进程间的资源共享(因此也称为共享库)
(3)将一些程序升级变得简单
(4)可以真正的做到链接载入完全由程序员在程序代码中控制(显示调用)
动态库的创建:(1)和静态库一样,然后属性-》配置属性-》常规-》配置类型
(2)win32工程-》dll
Windows系统下的执行文件格式是PE格式,动态库需要一个DllMain函数做初始化入口,通常用__declspecl(dllexport)关键字
static __declspec(dllexport) double add(double a,double b);
调用:
(1)若为第三方库-》VC++目录-》包含目录,库目录
链接器-》输入-》附加依赖项lib
dll考入到当前工作目录下
(2)若在同一个工程中-》通用属性-》框架和引用-》添加新引用
(3)显示调用若非类中函数-》VC++目录-》包含目录.h
.cpp中添加头文件typedef double (*Add)(double,double)函数申明
HINSTANCE stance = LoadLibraryA("../../Dll/StaticClassDll.dll")动态加载dll
Add add = (Add)GetProcAddress(stance,"AddDllGlobal")获取函数
double c = add(10,10)使用函数
FreeLibrary(stance)释放句柄
(4)显示调用若为类函数涉及到重载,则需要通过dll分析器进行名字粉碎从而和(3)中相同调用即可
AddM addm = (AddM)GetProcAddress(stance,"?AddXian@CXIANSHIDll@@QAENNN@Z");
example:
动态:
#ifdef XIANSHIDLL_EXPORTS
#define XIANSHIDLL_API __declspec(dllexport) //为了书写方便
#else
#define XIANSHIDLL_API __declspec(dllimport)
#endif
// 此类是从 XIANSHIDll.dll 导出的
extern "C" class XIANSHIDLL_API CXIANSHIDll { //extern "C" 为了可显示调用类中函数
public:
CXIANSHIDll(void);
double AddXian(double a,double b); //类中函数涉及重载需要进行函数粉碎,用dll分析器即可得到
};
extern XIANSHIDLL_API int nXIANSHIDll; //导出变量的接口
XIANSHIDLL_API int fnXIANSHIDll(void); //类外函数可直接进行显示调用