vc++动态链接库编程

上传人:xh****66 文档编号:57175104 上传时间:2018-10-19 格式:DOC 页数:62 大小:661KB
返回 下载 相关 举报
vc++动态链接库编程_第1页
第1页 / 共62页
vc++动态链接库编程_第2页
第2页 / 共62页
vc++动态链接库编程_第3页
第3页 / 共62页
vc++动态链接库编程_第4页
第4页 / 共62页
vc++动态链接库编程_第5页
第5页 / 共62页
点击查看更多>>
资源描述

《vc++动态链接库编程》由会员分享,可在线阅读,更多相关《vc++动态链接库编程(62页珍藏版)》请在金锄头文库上搜索。

1、1VC+VC+动态链接库编程之基础慨念动态链接库编程之基础慨念2005-10 作者:宋宝华 出处:天极网 http:/ 1.1.概论概论 先来阐述一下 DLL(Dynamic Linkable Library)的概念,你可以简单的把 DLL 看成一种仓库,它提供给你一些可以直接拿来用的变量、函数或类。在仓 库的发展史上经历了“无库静态链接库 动态链接库”的时代。静态链接库 与动态链接库都是共享代码的方式,如果采用静态链接库,则无论你愿不愿意, lib 中的指令都被直接包含在最终生成的 EXE 文件中了。但是若使用 DLL,该 DLL 不必被包含在最终 EXE 文件中,EXE 文件执行时可以“动

2、态”地引用和卸载 这个与 EXE 独立的 DLL 文件。静 态链接库和动态链接库的另外一个区别在于静 态链接库中不能再包含其他的动态链接库或者静态库,而在动态链接库中还可 以再包含其他的动态或静态链接库。对动态链接库,我们还需建立如下概念:(1)DLL 的编制与具体的编程语言及编译器无关只要遵循约定的 DLL 接口规范和调用方式,用各种语言编写的 DLL 都可以 相互调用。譬如 Windows 提供的系统 DLL(其中包括了 Windows 的 API),在任 何开发环境中都能被调用,不在乎其是 Visual Basic、Visual C+还是 Delphi。(2)动态链接库随处可见我们在 W

3、indows 目录下的 system32 文件夹中会看到 kernel32.dll、user32.dll 和 gdi32.dll,windows 的 大多数 API 都包含在这 些 DLL 中。kernel32.dll 中的函数主要处理内存管理和进程调度;user32.dll 中的函数主要控制用户界面; gdi32.dll 中的函数则负责图形方面的操作。一般的程序员都用过类似 MessageBox 的函数,其实它就包含在 user32.dll 这个动态链接库中。由此可见 DLL 对我们来说其实并不陌生。(3)VC 动态链接库的分类Visual C+支持三种 DLL,它们分别是 Non-MFC

4、DLL(非 MFC 动态库)、 MFC Regular DLL(MFC 规则 DLL)、MFC Extension DLL(MFC 扩展 DLL)。非 MFC 动态库不采用 MFC 类库结构,其导出函数为标准的 C 接口,能被非 MFC 或 MFC 编写的应用程序所调用;MFC 规则 DLL 包含一个继承自 CWinApp 的 类,但其无消息循环;MFC 扩展 DLL 采用 MFC 的动态链接版本创建,它只能被 用 MFC 类库所编写的应用程序所调用。2由于本文篇幅较长,内容较多,势必需要先对阅读本文的有关事项进行说 明,下面以问答形式给出。问:本文主要讲解什么内容?答:本文详细介绍了 DLL

5、 编程的方方面面,努力学完本文应可以对 DLL 有 较全面的掌握,并能编写大多数 DLL 程序。问:如何看本文?答:本文每一个主题的讲解都附带了源代码例程,可以随文下载(每个工 程都经 WINRAR 压缩)。所有这些例程都由笔者编写并在 VC+6.0 中调试通过。当然看懂本文不是读者的最终目的,读者应亲自动手实践才能真正掌握 DLL 的奥妙。问:学习本文需要什么样的基础知识?答:如果你掌握了 C,并大致掌握了 C+,了解一点 MFC 的知识,就可以轻 松地看懂本文。2.2.静态链接库静态链接库对静态链接库的讲解不是本文的重点,但是在具体讲解 DLL 之前,通过一 个静态链接库的例子可以快速地帮

6、助我们建立“库”的概念。图 1 建立一个静态链接库如图 1,在 VC+6.0 中 new 一个名称为 libTest 的 static library 工程3(单击此处下载本工程),并新建 lib.h 和 lib.cpp 两个文件,lib.h 和 lib.cpp 的源代码如下:/文件:lib.h#ifndef LIB_H #define LIB_H extern “C“ int add(int x,int y); /声明为 C 编译、连接方式 的外部函数 #endif/文件:lib.cpp#include “lib.h“ int add(int x,int y) return x + y; 编

7、译这个工程就得到了一个.lib 文件,这个文件就是一个函数库,它提供 了 add 的功能。将头文件和.lib 文件提交给用户后,用户就可以直接使用其中 的 add 函数了。标准 Turbo C2.0 中的 C 库函数(我们用来的 scanf、printf、memcpy、strcpy 等)就来自这种静态库。下面来看看怎么使用这个库,在 libTest 工程所在的工作区内 new 一个 libCall 工程。libCall 工程仅包含一个 main.cpp 文件,它演示了静态链接库 的调用方法,其源代码如下:#include #include “lib.h“ #pragma comment( li

8、b, “debuglibTest.lib“ ) /指定与静 态库一起连接int main(int argc, char* argv) printf( “2 + 3 = %d“, add( 2, 3 ) ); 静态链接库的调用就是这么简单,或许我们每天都在用,可是我们没有明 白这个概念。代码中#pragma comment( lib , “debuglibTest.lib“ )的 意思是指本文件生成的.obj 文件应与 libTest.lib 一起连接。如果不用 #pragma comment 指定,则可以直接在 VC+中设置,如图 2,依次选择4tools、options、directorie

9、s、library files 菜单或选项,填入库文件路径。 图 2 中加红圈的部分为我们添加的 libTest.lib 文件的路径。图 2 在 VC 中设置库文件路径这个静态链接库的例子至少让我们明白了库函数是怎么回事,它们是哪来 的。我们现在有下列模糊认识了:(1)库不是个怪物,编写库的程序和编写一般的程序区别不大,只是库不 能单独执行;(2)库提供一些可以给别的程序调用的东东,别的程序要调用它必须以某 种方式指明它要调用之。以上从静态链接库分析而得到的对库的懵懂概念可以直接引申到动态链接 库中,动态链接库与静态链接库在编写和调用上的不同体现在库的外部接口定 义及调用方式略有差异。 3.3

10、.库的调试与查看库的调试与查看在具体进入各类 DLL 的详细阐述之前,有必要对库文件的调试与查看方法 进行一下介绍,因为从下一节开始我们将面对大量的例子工程。由于库文件不能单独执行,因而在按下 F5(开始 debug 模式执行)或 CTRL+F5(运行)执行时,其弹出如图 3 所示的对话框,要求用户输入可执行文 件的路径来启动库函数的执行。这个时候我们输入要调用该库的 EXE 文件的路 径就可以对库进行调试了,其调试技巧与一般应用工程的调试一样。5图 3 库的调试与“运行”通常有比上述做法更好的调试途径,那就是将库工程和应用工程(调用库 的工程)放置在同一 VC 工作区,只对应用工程进行调试,

11、在应用工程调用库中 函数的语 句处设置断点,执行后按下 F11,这样就单步进入了库中的函数。第 2 节中的 libTest 和 libCall 工程就放在了同一工作区,其工程结构如图 4 所 示。图 4 把库工程和调用库的工程放入同一工作区进行调试上述调试方法对静态链接库和动态链接库而言是一致的。所以本文提供下 载的所有源代码中都包含了库工程和调用库的工程,这二者都被包含在一个工 作区内,这是笔者提供这种打包下载的用意所在。动态链接库中的导出接口可以使用 Visual C+的 Depends 工具进行查看, 让我们用 Depends 打开系统目录中的 user32.dll,看到了吧?红圈内的就

12、是几 个版本的 MessageBox 了!原来它真的在这里啊,原来它就在这里啊!6图 5 用 Depends 查看 DLL当然 Depends 工具也可以显示 DLL 的层次结构,若用它打开一个可执行文 件则可以看出这个可执行文件调用了哪些 DLL。好,让我们正式进入动态链接库的世界,先来看看最一般的 DLL,即非 MFC DLL。VC+VC+动态链接库编程之非动态链接库编程之非 MFCMFC DLLDLL 4.1 一个简单的 DLL 第 2 节给出了以静态链接库方式提供 add 函数接口的方法,接下来我们来 看看怎样用动态链接库实现一个同样功能的 add 函数。如图 6,在 VC+中 new

13、 一个 Win32 Dynamic-Link Library 工程 dllTest(单击此处下载本工程)。注意不要选择 MFC AppWizard(dll),因为 用 MFC AppWizard(dll)建立的将是第 5、6 节要讲述的 MFC 动态链接库。7图 6 建立一个非 MFC DLL在建立的工程中添加 lib.h 及 lib.cpp 文件,源代码如下:/* 文件名:lib.h */#ifndef LIB_H #define LIB_H extern “C“ int _declspec(dllexport)add(int x, int y); #endif/* 文件名:lib.cpp

14、*/#include “lib.h“ int add(int x, int y) return x + y; 与第 2 节对静态链接库的调用相似,我们也建立一个与 DLL 工程处于同一 工作区的应用工程 dllCall,它调用 DLL 中的函数 add,其源代码如下:#include #include typedef int(*lpAddFun)(int, int); /宏定义函数指针类型 int main(int argc, char *argv) HINSTANCE hDll; /DLL 句柄 lpAddFun addFun; /函数指针hDll = LoadLibrary(“Debugd

15、llTest.dll“);8if (hDll != NULL)addFun = (lpAddFun)GetProcAddress(hDll, “add“);if (addFun != NULL)int result = addFun(2, 3);printf(“%d“, result);FreeLibrary(hDll);return 0; 分析上述代码,dllTest 工程中的 lib.cpp 文件与第 2 节静态链接库版本 完全相同,不同在于 lib.h 对函数 add 的声明前面添加了 _declspec(dllexport)语句。这个语句的含义是声明函数 add 为 DLL 的导出函

16、数。DLL 内的函数分为两种:(1)DLL 导出函数,可供应用程序调用;(2) DLL 内部函数,只能在 DLL 程序使用,应用程序无法调用它们。而应用程序对本 DLL 的调用和对第 2 节静态链接库的调用却有较大差异, 下面我们来逐一分析。首先,语句 typedef int ( * lpAddFun)(int,int)定义了一个与 add 函数 接受参数类型和返回值均相同的函数指针类型。随后,在 main 函数中定义了 lpAddFun 的实例 addFun;其次,在函数 main 中定义了一个 DLL HINSTANCE 句柄实例 hDll,通过 Win32 Api 函数 LoadLibrary 动态加载了 DLL 模块并将 DLL 模块句柄赋给了 hDll;再次,在函数 main 中通过 Win32 Api 函数 GetProcAddress 得到了所加载 DLL 模块

展开阅读全文
相关资源
相关搜索

当前位置:首页 > 生活休闲 > 社会民生

电脑版 |金锄头文库版权所有
经营许可证:蜀ICP备13022795号 | 川公网安备 51140202000112号