Windows动态库的使用 part one

二.动态库程序
  这篇写得比较乱,看不懂的可以跳part 2,直接上图片的。
 
动态库的使用:

4.3.1 隐式链接

 动态库的隐式连接,需要动态库的 动态库文件".dll" , 动态库的LIB文件".lib" , 动态库的函数引入声明

1)头文件和函数原型

可以在函数原型的定义前,增加declspec(dllimport), 例如

_declspec(dllimport)  int   FuncName( ... );

如果库函数使用C格式导出,需要在函数定义增加 extern “C”

2)导入动态库的LIB文件

 #pragma comment(lib,"libzmq-v100-mt-4_0_4.lib");    //第二个参数是LIB文件的名字,可带地址

3)在程序中使用函数,像普通函数一样正常使用就好。

4)隐式链接的情况,DLL可以存放的路径:

(1)与执行文件中同一个目录下

(2)当前工作目录

(3)Windows目录

(4)Windows/System32目录

(5)Windows/System

(6)环境变量PATH指定目录

注意:高版本VC的配置文件

4.3.2 显式链接

1)定义函数指针类型

2)加载动态库

HMODULE LoadLibrary(

LPCTSTR lpFileName  //动态库文件名或全路径

); 返回DLL的实例句柄(HINSTANCE)

3)获取函数地址

FARPROC GetProcAddress(

HMODULE hModule,    //DLL句柄

LPCSTR lpProcName   //函数名称

); 成功返回函数地址

4)使用函数

5)卸载动态库

BOOL FreeLibrary(

HMODULE hModule   //DLL的实例句柄

);

4.3.3 两种链接方式对比

1)在库函数的定义不变情况下:

隐式链接,由于库函数地址是在程序编译链接时设置,所以当动态库变化后,使用程序需要重新编译链接。

显式链接,由于库函数地址是在程序执行时,动态的从库中查询,所以库变化后,不需要重新编译链接。

2)动态库加载

隐式链接,动态库是在程序启动时就被加载,当DLL不存在,程序无法启动

显式链接,动态库只在使用LoadLibrary函数,才会被加载。

/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

下面是一些概述,我自己的笔记做得比较乱,只有自己才看得懂了。如果不想看的,可以直接看part 2的例子。

/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

1.动态库特点

1)运行时独立存在

2)不会链接到执行程序

3)使用时加载

与静态库的比较:

1)由于静态库是将代码嵌入到使用程序中,多个程序使用时,会有多份代码,所以代码体积会增大。动态库的代码只需要存在一份,其他程序通过函数地址使用,所以代码体积小。

2)静态库发生变化后,新的代码需要重新链接嵌入到执行程序中。动态库发生变化后,如果库中函数的定义(或地址)未变化,其他使用DLL的程序不需重新链接。

2.动态库的创建

1 ) 建立项目

2 ) 添加库程序

3 ) 库程序导出 - 提供给使用者库中的函数等信息。

3.动态库的使用

3.1 隐式链接

3.2 显式链接

4.动态库的函数

4.1 实现动态库的函数

4.2 库函数的导出

1) C++的导出

使用 _declspec(dllexport) 导出函数

注意:动态库编译链接后,也会有LIB文件,是作为动态库函数映射使用,与静态库不完全相同。

2)C的导出方式     extern “C” _declspec(dllexport) int Sub(...);

3)模块定义文件 .def

例如:

LIBRARY DLLFunc //库

EXPORTS         //库导出表

DLL_Mul  @1  //导出的函数


4.3 库函数的使用

4.3.1 隐式链接

1)头文件和函数原型

可以在函数原型的定义前,增加declspec(dllimport), 例如

_declspec(dllimport)  int   FuncName( ... );

如果库函数使用C格式导出,需要在函数定义增加 extern “C”

2)导入动态库的LIB文件

 #pragma comment(lib,"libzmq-v100-mt-4_0_4.lib");    //第二个参数是LIB文件的名字,可带地址

3)在程序中使用函数,像普通函数一样正常使用就好。

4)隐式链接的情况,DLL可以存放的路径:

(1)与执行文件中同一个目录下

(2)当前工作目录

(3)Windows目录

(4)Windows/System32目录

(5)Windows/System

(6)环境变量PATH指定目录

注意:高版本VC的配置文件

4.3.2 显式链接

1)定义函数指针类型

2)加载动态库

HMODULE LoadLibrary(

LPCTSTR lpFileName  //动态库文件名或全路径

); 返回DLL的实例句柄(HINSTANCE)

3)获取函数地址

FARPROC GetProcAddress(

HMODULE hModule,    //DLL句柄

LPCSTR lpProcName   //函数名称

); 成功返回函数地址

4)使用函数

5)卸载动态库

BOOL FreeLibrary(

HMODULE hModule   //DLL的实例句柄

);

4.3.3 两种链接方式对比

1)在库函数的定义不变情况下:

隐式链接,由于库函数地址是在程序编译链接时设置,所以当动态库变化后,使用程序需要重新编译链接。

显式链接,由于库函数地址是在程序执行时,动态的从库中查询,所以库变化后,不需要重新编译链接。

2)动态库加载

隐式链接,动态库是在程序启动时就被加载,当DLL不存在,程序无法启动

显式链接,动态库只在使用LoadLibrary函数,才会被加载。


DLL中类的使用

1.DLL中类的导出

在类名称前增加 _declspec(dllexport) 定义,例如:

class _declspec(dllexport) CMath {

...

};

通常使用预编译开关切换类的导入导出定义,例如:

#ifdef DLLCLASS_EXPORTS

#define EXT_CLASS _declspec(dllexport)//DLL

#else

#define EXT_CLASS _declspec(dllimport)//使用者

#endif

class EXT_CLASS CMath{

...

};

2.使用DLL中的类

1 导入DLL的LIb

2 类的定义

3 使用类

3.动态库的程序入口

入口程序不是DLL必须的。常用于DLL内部初始化或善后处理。

BOOL WINAPI DllMain(

HINSTANCE hinstDLL, //动态库实例句柄

DWORD fdwReason,    //被调用的原因

LPVOID lpvReserved   //保留值

); 返回TRUE,表示动态库加载成功。

动态库的加载或卸载时会被调用。例如:使用LoadLibrary或FreeLibrary时会被调用。

附:

制作动态库:
1.新建选择倒数第2个:Win32 Dynamic-Link Library
2.函数导出(生成lib文件以调用):_declspec(dllexport)
3.Project/Settings/Links 设置 .dll 和 .ilk 的位置 复制.lib到指定存放位置

4.使用库文件:
C语言:
extern "C"_declspec(dllimport)函数;
#pragma comment(lib,"../lib/Cdll.lib")
C++语言:
_declspec(dllimport)函数;
#pragma comment(lib,"../lib/CPPdll.lib")

1.声明导出:
将函数的偏移地址 导到了 和dll配套生成lib文件中
2.模块文件
将函数的偏移地址 导到了 dll文件 和 lib文件 各一份

使用动态库
1.隐式链接
将dll文件内容导到内存的过程不需要程序员负责。
链接器从lib文件中获取函数的偏移地址。
2.显示链接
将dll文件中内容导到内存的过程需要程序员自己负责。

DLL中类的使用:
1.DLL中类的导出
在类名称前增加 _declspec(dllexport) 定义,例如:

class _declspec(dllexport) CMath {
...
};
2.通常使用预编译开关切换类的导入导出定义,例如:

#ifdef DLLCLASS_EXPORTS

#define EXT_CLASS _declspec(dllexport)//DLL

#else

#define EXT_CLASS _declspec(dllimport)//使用者

#endif

class EXT_CLASS CMath{
...
};
******* *******
// 把这些写在头文件中,用户只要加上头文件就行了

实现例子晚些再补上,可以参考这个:http://blog.sina.com.cn/s/blog_6fb3686501011ymn.html

上一篇:Java Socket 编程指南


下一篇:URL路由系统-命名空间