动态库dll的调用方式-显式链接
这里有个疑问,VS2017生成dll时为何会同时生成一个lib文件?
静态库生成lib,动态库也生成lib。
动态库生成的那个lib一般管它叫"导入库",这样的lib在大多数情况下要比静态库的lib小,
里面不包含涉及到的函数的具体代码,里面只包含"这个函数在什么dll里面叫什么名字"这样的信息。即函数的声明
你平时编程比如windows.h里的那些api,都是通过这种方式链接的。
比如弹出消息框的MessageBoxA,它在user32.lib中,但是user32.lib却没有包含MessageBoxA的具体代码,
它只写明了这个函数在user32.dll中,名字叫做MessageBoxA。生成exe以后,windows在运行这个exe里的代码之前,
会先载入user32.dll,然后在里面查找MessageBoxA,随后把MessageBoxA所在的地址写入exe指定的位置(一般是导入表)。
dll要调用这个函数的时候,就从导入表获取函数地址,然后调用过去。
方法2:显式链接---需要DLL(不需要.lib,.h文件)
注意:显示链接,导出DLL的时候,采用extern “C”的方式,
而不采用_declspec(dllexport),因为_declspec(dllexport)会有一个”名字改编”的问题(采用了_cdecl调用规约的C++编译方式)。
如库文件对应的头文件如下 #ifndef DLLTEST_H #defineDLLTEST_H //该宏完成在dll项目内部使用__declspec(dllexport)导出 //在dll项目外部使用时,用__declspec(dllimport)导入 //宏DLL_EXPORTS在.cpp中定义 #ifdefDLL_EXPORTS #defineDLL_EXPORTS extern"C"_declspec(dllexport) #else #define DLL_EXPORTS extern"C"_declspec(dllimport) #endif //函数声明 DLL_EXPORTSint Add(int a,int b); DLL_EXPORTSint Sub(int a,int b); DLL_EXPORTSint Divide(int a,int b); #endif//DLLTEST_H
测试代码: typedef int(*funAdd)(int a, int b); typedef int(fun *Sub)(int a, int b); HINSTANCE hDLL; funAdd Add;//函数指针 hDLL =LoadLibrary(_T("D:/DLLTest.dll"));//加载动态链接库DLLTest.dll文件; Add = (funAdd)GetProcAddress(hDLL,"Add"); int result = Add(5, 8); printf("5+8:%d\n",result); FreeLibrary(hDLL);//卸载.dll文件;
以下是采用__declspec(dllexport)方式导出,不推荐这种方式需要注意函数名的书写问题! 现在DLLTest.h中有函数(采用__declspec(dllexport)方式导出) #ifndef DLLTEST_H #defineDLLTEST_H //该宏完成在dll项目内部使用__declspec(dllexport)导出 //在dll项目外部使用时,用__declspec(dllimport)导入 //宏DLL_EXPORTS在.cpp中定义 #ifdefDLL_EXPORTS #defineDLL_EXPORTS __declspec(dllexport) #else #define DLL_EXPORTS __declspec(dllimport) #endif intDLL_EXPORTS Add(int a,int b); intDLL_EXPORTS Sub(int a,int b); intDLL_EXPORTS Divide(int a,int b); #endif//DLLTEST_H
测试代码 typedef int(*funAdd)(int a, int b); typedef int(*funSub)(int a, int b); HINSTANCE hDLL; funAdd Add;//函数指针 //也可以采用hDLL =LoadLibrary(_T("D:/DLLTest.dll")); hDLL =LoadLibraryA(("D:/DLLTest.dll"));//加载动态链接库DLLTest.dll文件; Add = (funAdd)GetProcAddress(hDLL,"?Add@@YAHHH@Z");//!!!!获取函数地址 int result = Add_(5, 8); printf("5+8:%d\n",result); FreeLibrary(hDLL);//卸载.dll文件;
这里需要注意的是:GetProcAddress(hDLL,"函数名");中的函数名要是DLL中的函数名,这个函数名可以用PE Explorer软件查看
由于采用的是VC++处理函数名方式,所以
"?Add@@YAHHH@Z";// GetProcAddress(hDLL,"函数名");
而不是简单的“Add”,因为DLL中的函数名是结果VC++方式处理过的函数名
显式调用,最好采用extern “C”的方式导出DLL
所以显式隐式链接方式的时候,只加载需要的DLL,在附加依赖项中,只添加需要的DLL 对于的lib,不要多加
否则会造成
1.加大程序启动时间
2.内存浪费

浙公网安备 33010602011771号