2024年4月30日发(作者:)

1、概述

先来阐述一下DLL(Dynamic Linkable Library)的概念,你可以简单的

把DLL看成一种仓库,它提供给你一些可以直接拿来用的变量、函数或类。在仓

库的发展史上经历了“无库-静态链接库-动态链接库”的时代。

目前以lib后缀的库有两种,一种为静态链接库(Static Libary,以下

简称“静态库”),另一种为动态连接库(DLL,以下简称“动态库”)的导入库

(Import Libary,以下简称“导入库”)。

静态链接库与动态链接库都是共享代码的方式,如果采用静态链接库,

则无论你愿不愿意,lib中的指令都被直接包含在最终生成的EXE文件中了。但

是若使用DLL,该DLL不必被包含在最终 EXE文件中,EXE文件执行时可以“动

态”地引用和卸载这个与EXE独立的DLL文件。静态链接库和动态链接库的另外

一个区别在于静态链接库中不能再包含其他的动态链接库或者静态库,而在动态

链接库中还可以再包含其他的动态或静态链接库。

2、静态库

对静态链接库的讲解不是本文的重点,但是在具体讲解DLL之前,通过

一个静态链接库的例子可以快速地帮助我们建立“库”的概念。

在VC++6.0中new一个名称为libTest的static library工程(单击此

处下载本工程附件),并新建lib.h和两个文件,lib.h和的

源代码如下:

//文件:lib.h

#ifndef LIB_H

#define LIB_H

extern "C" int add(int x,int y); //声明为C编译、连接方式的外部函

#endif

//文件:

#include "lib.h"

int add(int x,int y)

{

return x + y;

}

编译这个工程就得到了一个.lib文件,这个文件就是一个函数库,它提供

了add的功能。将头文件和.lib文件提交给用户后,用户就可以直接使用其中

的add函数了。

下面来看看怎么使用这个库,在libTest工程所在的工作区内new一个libCall

工程。libCall工程仅包含一个文件,它演示了静态链接库的调用方

法,其源代码如下:

#include

#include "../lib.h"

#pragma comment( lib, "..//debug//" ) //指定与静态库一起

连接

int main(int argc, char* argv[])

{

printf( "2 + 3 = %d", add( 2, 3 ) );

}

静态链接库的调用就是这么简单,或许我们每天都在用,可是我们没有明白

这个概念。代码中#pragma comment( lib , "..//debug//" )的意

思是指本文件生成的.obj文件应与一起连接。

如果不用#pragma comment指定,则可以直接在VC++中设置,如图2,

依次选择tools、options、directories、library files菜单或选项,填入库

文件路径。图2中加红圈的部分为我们添加的文件的路径。

这个静态链接库的例子至少让我们明白了库函数是怎么回事,它们是哪来的。

我们现在有下列模糊认识了:

(1)库不是个怪物,编写库的程序和编写一般的程序区别不大,只是库不

能单独执行;

(2)库提供一些可以给别的程序调用的东东,别的程序要调用它必须以某

种方式指明它要调用之。

以上从静态链接库分析而得到的对库的懵懂概念可以直接引申到动态链接

库中,动态链接库与静态链接库在编写和调用上的不同体现在库的外部接口定义

及调用方式略有差异。

3、 库的调试与查看

由于库文件不能单独执行,因而在按下F5(开始debug模式执行)或

CTRL+F5(运行)执行时,其弹出提示对话框,要求用户输入可执行文件的路径

来启动库函数的执行。这个时候我们输入要调用该库的EXE文件的路径就可以对

库进行调试了,其调试技巧与一般应用工程的调试一样。

4、动态库

动态链接库(Dynamic-link library,DLL)是Microsoft Windows组件

模型的核心。动态链接库是一个可执行模块,包含一些独立的文件,但是与应用

程序不同,动态链接库没有自己的消息循环,且不能单独执行,需要通过应用程

序或者其他动态链接库的调用才能执行。动态链接通常是封装了某些功能的函数

集合,用来提供给其他应用程序调用。

通常有比上述做法更好的调试途径,那就是将库工程和应用工程(调用

库的工程)放置在同一VC工作区,只对应用工程进行调试,在应用工程调用库

中函数的语句处设置断点,执行后按下F11,这样就单步进入了库中的函数。

动态链接库在使用时,一般都提供引入文件(.lib)和动态链接库文件

(.dll)。其中,引入库文件提供了调用函数的位置信息,而动态链接库文件才

真正包含函数的源代码。动态库一般会有对应的导入库,方便程序静态载入动态

链接库,否则你可能就需要自己LoadLibary调入DLL文件,然后再手工

GetProcAddress获得对应函数了。有了导入库,你只需要链接导入库后按照头

文件函数接口的声明调用函数就可以了。导入库和静态库 的区别很大,他们实

质是不一样的东西。静态库本身就包含了实际执行代码、符号表等等 ,而对于

导入库而言,其实际的执行代码位于动态库中,导入库只包含了地址符号表等,

确保程序找到对应函数的一些基本地址信息 。

对动态链接库,我们还需建立如下概念:

(1)DLL 的编制与具体的编程语言及编译器无关

只要遵循约定的DLL接口规范和调用方式,用各种语言编写的DLL都可以相

互调用。譬如Windows提供的系统DLL(其中包括了Windows的API),在任何

开发环境中都能被调用,不在乎其是Visual Basic、Visual C++还是Delphi。

(2)动态链接库随处可见

我们在Windows目录下的system32文件夹中会看到、

和,windows的大多数API都包含在这些DLL中。

中的函数主要处理内存管理和进程调度;中的函数主

要控制用户界面;中的函数则负责图形方面的操作。

一般的程序员都用过类似MessageBox的函数,其实它就包含在

这个动态链接库中。由此可见DLL对我们来说其实并不陌生。

(3)VC动态链接库的分类

Visual C++支持三种DLL,它们分别是Non-MFC DLL(非MFC动态库)、MFC

Regular DLL(MFC规则DLL)、MFC Extension DLL(MFC扩展DLL)。

非MFC动态库不采用MFC类库结构,其导出函数为标准的C接口,能被

非MFC或MFC编写的应用程序所调用;MFC规则DLL 包含一个继承自CWinApp

的类,但其无消息循环;MFC扩展DLL采用MFC的动态链接版本创建,它只能被

用MFC类库所编写的应用程序所调用。

4.1、非MFC DLL

1、打开File —> New —> Project选项,选择Win32 Dynamic-Link Library —>sample project —>

工程名:DllDemo

2、新建一个.h文件DllDemo.h

#ifdef DllDemo_EXPORTS

#define DllAPI __declspec(dllexport)

#else

#define DllAPI __declspec(dllimport)

extern "C" //原样编译

{

DllAPI int __stdcall Max(int a,int b); //__stdcall使非C/C++语言内能

够调用API

}

#endif

3、在文件中导入DllDemo.h文件,并实现Max(int,int)函数

#include "DllDemo.h"

DllAPI int __stdcall Max(int a,int b)

{

if(a==b)

return NULL;

else if(a>b)

return a;

else

return b;

}

4、编译程序生成动态连接库