VC++动态链接库创建和调用全过程详解

上传人:桔**** 文档编号:500758619 上传时间:2022-10-19 格式:DOC 页数:24 大小:448.50KB
返回 下载 相关 举报
VC++动态链接库创建和调用全过程详解_第1页
第1页 / 共24页
VC++动态链接库创建和调用全过程详解_第2页
第2页 / 共24页
VC++动态链接库创建和调用全过程详解_第3页
第3页 / 共24页
VC++动态链接库创建和调用全过程详解_第4页
第4页 / 共24页
VC++动态链接库创建和调用全过程详解_第5页
第5页 / 共24页
点击查看更多>>
资源描述

《VC++动态链接库创建和调用全过程详解》由会员分享,可在线阅读,更多相关《VC++动态链接库创建和调用全过程详解(24页珍藏版)》请在金锄头文库上搜索。

1、.1.概论先来阐述一下DLL(Dynamic Linkable Library)的概念,你可以简单的把DLL看成一种仓库,它提供给你一些可以直接拿来用的变量、函数或类。在仓库的发展史上经历了“无库静态库动态库的时代。静态库与动态库都是共享代码的方式,如果采用静态库,那么无论你愿不愿意,lib中的指令都被直接包含在最终生成的EXE文件中了。但是假设使用DLL,该DLL不必被包含在最终EXE文件中,EXE文件执行时可以“动态地引用和卸载这个与EXE独立的DLL文件。静态库和动态库的另外一个区别在于静态库中不能再包含其他的动态库或者静态库,而在动态库中还可以再包含其他的动态或静态库。对动态库,我们还

2、需建立如下概念:1DLL 的编制与具体的编程语言及编译器无关只要遵循约定的DLL接口规和调用方式,用各种语言编写的DLL都可以相互调用。譬如Windows提供的系统DLL其中包括了Windows的API,在任何开发环境中都能被调用,不在乎其是Visual Basic、Visual C+还是Delphi。2动态库随处可见我们在Windows目录下的system32文件夹中会看到kernel32.dll、user32.dll和gdi32.dll,windows的大多数API都包含在这些DLL中。kernel32.dll中的函数主要处理存管理和进程调度;user32.dll中的函数主要控制用户界面;

3、gdi32.dll中的函数那么负责图形方面的操作。一般的程序员都用过类似MessageBox的函数,其实它就包含在user32.dll这个动态库中。由此可见DLL对我们来说其实并不陌生。(3)VC动态库的分类Visual C+支持三种DLL,它们分别是Non-MFC DLL非MFC动态库、MFC Regular DLLMFC规那么DLL、MFC Extension DLLMFC扩展DLL。非MFC动态库不采用MFC类库结构,其导出函数为标准的C接口,能被非MFC或MFC编写的应用程序所调用;MFC规那么DLL 包含一个继承自CWinApp的类,但其无消息循环;MFC扩展DLL采用MFC的动态版

4、本创建,它只能被用MFC类库所编写的应用程序所调用。由于本文篇幅较长,容较多,势必需要先对阅读本文的有关事项进行说明,下面以问答形式给出。问:本文主要讲解什么容?答:本文详细介绍了DLL编程的方方面面,努力学完本文应可以对DLL有较全面的掌握,并能编写大多数DLL程序。问:如何看本文?答:本文每一个主题的讲解都附带了源代码例程,可以随文下载每个工程都经WINRAR压缩。所有这些例程都由笔者编写并在VC+6.0中调试通过。当然看懂本文不是读者的最终目的,读者应亲自动手实践才能真正掌握DLL的奥妙。问:学习本文需要什么样的基础知识?答:如果你掌握了C,并大致掌握了C+,了解一点MFC的知识,就可以

5、轻松地看懂本文。2.静态库对静态库的讲解不是本文的重点,但是在具体讲解DLL之前,通过一个静态库的例子可以快速地帮助我们建立“库的概念。图1 建立一个静态库如图1,在VC+6.0中new一个名称为libTest的static library工程单击此处下载本工程附件,并新建lib.h和lib.cpp两个文件,lib.h和lib.cpp的源代码如下:/文件:lib.h#ifndef LIB_H#define LIB_Hextern C int add(int x,int y);/声明为C编译、连接方式的外部函数#endif/文件:lib.cpp#include lib.hint add(int

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

7、Test.lib ) /指定与静态库一起连接int main(int argc, char* argv)printf( 2 + 3 = %d, add( 2, 3 ) );静态库的调用就是这么简单,或许我们每天都在用,可是我们没有明白这个概念。代码中#pragma ment( lib , .debuglibTest.lib )的意思是指本文件生成的.obj文件应与libTest.lib一起连接。如果不用#pragma ment指定,那么可以直接在VC+中设置,如图2,依次选择tools、options、directories、library files菜单或选项,填入库文件路径。图2中加红圈的

8、部分为我们添加的libTest.lib文件的路径。图2 在VC中设置库文件路径这个静态库的例子至少让我们明白了库函数是怎么回事,它们是哪来的。我们现在有以下模糊认识了:1库不是个怪物,编写库的程序和编写一般的程序区别不大,只是库不能单独执行;2库提供一些可以给别的程序调用的东东,别的程序要调用它必须以某种方式指明它要调用之。以上从静态库分析而得到的对库的懵懂概念可以直接引申到动态库中,动态库与静态库在编写和调用上的不同表达在库的外部接口定义及调用方式略有差异。3.库的调试与查看在具体进入各类DLL的详细阐述之前,有必要对库文件的调试与查看方法进行一下介绍,因为从下一节开始我们将面对大量的例子工

9、程。由于库文件不能单独执行,因而在按下F5开始debug模式执行或CTRL+F5运行执行时,其弹出如图3所示的对话框,要求用户输入可执行文件的路径来启动库函数的执行。这个时候我们输入要调用该库的EXE文件的路径就可以对库进行调试了,其调试技巧与一般应用工程的调试一样。图3 库的调试与“运行通常有比上述做法更好的调试途径,那就是将库工程和应用工程调用库的工程放置在同一VC工作区,只对应用工程进行调试,在应用工程调用库中函数的语句处设置断点,执行后按下F11,这样就单步进入了库中的函数。第2节中的libTest和libCall工程就放在了同一工作区,其工程结构如图4所示。图4把库工程和调用库的工程

10、放入同一工作区进行调试上述调试方法对静态库和动态库而言是一致的。所以本文提供下载的所有源代码中都包含了库工程和调用库的工程,这二者都被包含在一个工作区,这是笔者提供这种打包下载的用意所在。动态库中的导出接口可以使用Visual C+的Depends工具进行查看,让我们用Depends打开系统目录中的user32.dll,看到了吧?红圈的就是几个版本的MessageBox了!原来它真的在这里啊,原来它就在这里啊!图5用Depends查看DLL当然Depends工具也可以显示DLL的层次结构,假设用它打开一个可执行文件那么可以看出这个可执行文件调用了哪些DLL。好,让我们正式进入动态库的世界,先来

11、看看最一般的DLL,即非MFC DLL(待续.)上节给大家介绍了静态库与库的调试与查看动态库(DLL)编程深入浅出(一),本节主要介绍非MFC DLL。4.非MFC DLL4.1一个简单的DLL第2节给出了以静态库方式提供add函数接口的方法,接下来我们来看看怎样用动态库实现一个同样功能的add函数。如图6,在VC+中new一个Win32 Dynamic-Link Library工程dllTest单击此处下载本工程附件。注意不要选择MFC AppWizard(dll),因为用MFC AppWizard(dll)建立的将是第5、6节要讲述的MFC 动态库。图6 建立一个非MFC DLL在建立的工

12、程中添加lib.h及lib.cpp文件,源代码如下:/* 文件名:lib.h*/#ifndef LIB_H#define LIB_Hextern C int _declspec(dllexport)add(int x, int y);#endif/* 文件名:lib.cpp*/#include lib.hint add(int x, int y)return x + y;与第2节对静态库的调用相似,我们也建立一个与DLL工程处于同一工作区的应用工程dllCall,它调用DLL中的函数add,其源代码如下:#include #include typedef int(*lpAddFun)(int,

13、 int); /宏定义函数指针类型int main(int argc, char *argv)HINSTANCE hDll; /DLL句柄 lpAddFun addFun; /函数指针hDll = LoadLibrary(.DebugdllTest.dll);if (hDll != NULL)addFun = (lpAddFun)GetProcAddress(hDll, add);if (addFun != NULL)int result = addFun(2, 3);printf(%d, result);FreeLibrary(hDll);return 0;分析上述代码,dllTest工程中

14、的lib.cpp文件与第2节静态库版本完全相同,不同在于lib.h对函数add的声明前面添加了_declspec(dllexport)语句。这个语句的含义是声明函数add为DLL的导出函数。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模块中函数add的地址并赋给了addFun。经由函数指针addFun进行了对DLL中add函数的调用;最后

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

当前位置:首页 > 建筑/环境 > 施工组织

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