八宝书库 > 文学其他电子书 > C语言实例教程(PDF格式) >

第83部分

C语言实例教程(PDF格式)-第83部分

小说: C语言实例教程(PDF格式) 字数: 每页4000字

按键盘上方向键 ← 或 → 可快速上下翻页,按键盘上的 Enter 键可回到本书目录页,按键盘上方向键 ↑ 可回到本页顶部!
————未阅读完?加入书签已便下次继续阅读!




  用的函数都具有相同的接口,这样就可以通过简单地更换DLL来实 

  现多语言支持。  



然而,使用DLL也有其不足之处。最典型的就是应用程序在运行时必 


…………………………………………………………Page 663……………………………………………………………

须要有相应的DLL的支持。另外,使用DLL也增大了程序运行的开销, 

好在这种额外的开销对于大多数应用程序的影响并不明显,我们也只 

是在某些对运行速度要求苛刻的特殊场合,才不得不考虑这一点。  



Visual C++ 5。0支持多种DLL,包括:  



  l 非MFC DLL  



  l 静态链接到MFC的常规DLL  



  l 动态链接到MFC的常规DLL  



  l MFC扩展DLL  



其中非MFC DLL (non…MFC DLL )内部不使用MFC,调用非MFC DLL提供 

的导出函数的可执行程序可以使用MFC,也可以不使用MFC。一般来 

                                     standard  C  

说,非MFC     DLL的导出函数都使用标准的C接口 ( 

interface)。  



其余三种DLL的内部都使用了MFC。顾名思义,静态链接到MFC的常规 

DLL (regular DLL statically linking to MFC)和动态链接到MFC 

的常规DLL (regular DLL dynamically linking to MFC)的区别在 

于一个使用的是MFC的静态链接库,而另一个使用的是MFC的DLL。这 

和一般的MFC应用程序的情况是很类似的。  



MFC扩展DLL一般用来提供派生于MFC的可重用的类,以扩展已有的MFC 

类库的功能。MFC扩展DLL使用MFC的动态链接版本。只有使用MFC动态 

链接的可执行程序 (无论是EXE还是DLL)才能访问MFC扩展DLL。MFC 

扩展DLL的另一个有用的功能是它可以在应用程序和它所加载的MFC扩 

展DLL之间传递MFC和MFC派生对象的指针。在其它情况下,这样做是 

可能导致问题的。  



  l 注意:  



  l 只有Visual C++ 5。0的专业版和企业版才支持到MFC的静态链接。  



  l 静态链接到MFC的常规DLL过去的USRDLL有着相同的特性,同样 

   的,MFC扩展DLL和过去的AFXDLL有着相同的特性。在Visual C++  

   5。0中已不再使用USRDLL和AFXDLL这两个术语。  



如何选择所应该使用的DLL的类型呢?我们可以从以下几个方面来考 

虑:  


…………………………………………………………Page 664……………………………………………………………

  l 相比使用了MFC的DLL而言,非MFC DLL显得更为短小精悍。因此, 

   如果DLL不需要使用MFC,那么使用非MFC DLL是一个很好的选择, 

   它将显著地节省磁盘和内存空间。同时,无论应用程序是否使用 

   了MFC,都可以调用非MFC DLL中所导出的函数。  



  l 如果需要创建使用了MFC的DLL,并希望MFC和非MFC应用程序都能 

   使用所创建的DLL,那么可以选择的范围包括静态链接到MFC的常 

   规DLL和动态链接到MFC的常规DLL。动态链接到MFC的常规DLL比较 

   短小,因此可以节省磁盘和内存,但是,在分发动态链接到MFC的 

   常规DLL时,必须同时分发MFC的支持DLL,如MFCx0。DLL和 

   MSVCRT。DLL等。而使用静态链接到MFC的常规DLL则不存在这种问 

   题。  



  l 如果希望在DLL中实现从MFC派生的可重用的类,或者是希望在应 

   用程序和DLL之间传递MFC的派生对象时,必须选择MFC扩展DLL。  



             第二节 创建和使用动态链接库  



本节以非MFC  DLL为例来讲解DLL的结构和导出方法,并介绍创建和使 

用DLL的方法和步骤。  



13。2。1 DLL的结构和导出方式  



DLL文件和EXE文件都属于可执行文件,不同的是DLL文件包括了一个 

导出表,导出表中给出了可以从DLL中导出的所有函数的名字。外部 

可执行程序只能访问包括在DLL的导出表中的函数,DLL中的其它函数 

是私有的,不能为外部可执行程序所访问。  



可以使用Visual              C++提供的DUMPBIN实用程序 (可以在 

DevStudioVCbin 目录下找到这个工具)来查看一个DLL文件的结 

构。举一个例子,如果需要查看DLL文件msgbox。dll (我们将在本小 

节的后续内容中创建该DLL)的导出表,可以在命令提示符下键入下 

面的命令:  



》dumpbin /exports msgbox。dll  



运行结果如下:  



Microsoft (R) COFF Binary File Dumper Version 5。00。7022  



Copyright (C) Microsoft Corp 1992…1997。 All rights reserved。  



Dump of file msgbox。dll  


…………………………………………………………Page 665……………………………………………………………

File Type: DLL  



Section contains the following Exports for MSGBOX。dll  



0 characteristics  



351643C3 time date stamp Mon Mar 23 19:13:07 1998  



0。00 version  



1 ordinal base  



1 number of functions  



1 number of names  



ordinal hint name  



1 0 MsgBox (00001000)  



Summary  



7000 。data  



1000 。idata  



2000 。rdata  



2000 。reloc  



17000 。text  



由上面的结果得知,msgbox。dll中仅包括了一个导出函数MsgBox()。  



   l 注意:  



   l 仅仅知道导出函数的名称并不足以从DLL 中导出该函数。若在应用 

     程序中使用显式链接 (link explicitly),至少还应该知道导出 

     函数的返回值的类型以及所传递给导出函数的参数的个数、顺序 

     和类型;若使用隐含链接 (link implicitly),必须有包括导出 

     函数 (或类)的定义的头文件 (。H文件)和引入库 (import  

     library,。LIB文件),这些文件是由DLL的创建者所提供的。关 

     于显式链接和隐含链接,将在本章的 “13。2。2 链接应用程序到 

     DLL”小节中讲述。  



从DLL中导出函数有两种方法:  



   l 在创建DLL时使用模块定义 (module DEFinition,。DEF)文件。  


…………………………………………………………Page 666……………………………………………………………

  l 在定义函数时使用关键字__declspec(dllexport)。  



下面我们通过一个简单的例子来分别说明两种方法的使用。在这个例 

子中,我们将创建一个只包括一个函数MsgBox() 的DLL,函数MsgBox 

()用来显示一个消息框,它和Win32   API函数MessageBox()的功能是 

一样,只不过在函数MsgBox()中,不需要指定消息的父窗口,而且可 

以缺省其它所有的参数。  



   (1) 使用模块定义文件  



模块定义文件是一个文本文件,它包括了一系列的模块语句,这些语 

句用来描述DLL的各种属性,典型的,模块语句定义了DLL中所导出的 

函数的名称和顺序值。  



在讲解模块定义文件之前,我们先创建一个Win32                   Dynamic…Link  

Library工程。  



1。 在Microsoft Developer Studio中选择File菜单下的New命令,在 

Projects选项卡中选择Win32 Dynamic…Link Library,并为工程取一 

个名字,如msgbox。单击OK后,Visual  C++创建一个Win32  DLL的空 

白工程,必须手动的将所需要的文件添加到工程中。  



2。  单击Project菜单下的Add  To  Project子菜单下的New命令,在 

Files选项卡中选择Text   File,在File文本框中输入DEF文件名,如 

msgbox。def。  



3。   双击Workspace窗口的FileView选项卡中的msgbox。def节点,在 

msgbox。def文件中输入下面的内容:  



LIBRARY MSGBOX  



DESCRIPTION 〃一个DLL的简单例子〃  



EXPORTS  



MsgBox @1  



在DEF文件中的第一条语句必须是LIBRARY语句,该语句表明该DEF文 

件属于一个DLL,在LIBRARY之后是DLL的名称,这个名称在链接时将 

放到DLL的引入库中。  



EXPORTS语句下列出了DLL的所有导出函数以及它们的顺序值。函数的 

顺序值不是必须的,在指定导出函数的顺序值时,我们在函数名后跟 


…………………………………………………………Page 667……………………………………………………………

上一个@符号和一个数字,该数字即导出函数的顺序值。如果在DEF中 

指定了顺序值,它必须不小于1,且不大于DLL中所有导出函数的数 

目。  



DESCRIPTION语句是可选的,它简单的说明了DLL的用途。  



4。  下一步是向工程中添加一个头文件,它定义了DLL中的函数的返回 

值的类型和参数的个数、顺序和类型。  



单击菜单项Project|Add To Project|New。。。,在Files选项卡下选择 

C/C++   Header   File,在File文本框中指定头文件名,如msgbox。h 

 (可以省略后缀名。h)。  



在头文件中输入如下的内容:  



#include   



extern 〃C〃 int MsgBox(  



// 消息框的文本  



LPCTSTR lpText=〃虽然这个例子有一些幼稚,但它工作得非常的好 !〃;  



// 消息框的标题  



LPCTSTR lpCaption=〃一个简单的例子〃;  



// 消息框的样式  



UINT uType=MB_OK);  



请注意函数定义前的关键字extern   〃C〃,这是由于我们使用了C++语 

言来开发DLL,为了使C语言模块能够访问该导出函数,我们应该使用 

C链接来代替C++链接。否则,C++编译器将使用C++的类型安全命名和 

调用协议,这在使用C调用该函数时就会遇上问题。在本例中并不需 

要考虑到这个问题,因为我们在开发DLL和应用程序时都是使用C++, 

但我们仍然强烈建议使用extern   〃C〃,以保证在使用C编写的程序调 

用该DLL的导出函数不会遇上麻烦,在本章后面的内容中我们还会讨 

论到这个问题。  



5。  下面要做的事是向工程中添加一个C++源文件,在该文件中实现函 

数MsgBox()。  



仿照上面的过程,单击菜单项Project|Add  To  Project|New。。。,在 

Files选项卡下选择C++  Source  File,在File文本框中指定源文件 


…………………………………………………………Page 668……………………………………………………………

名,如msgbox。cpp。  



在msgbox。cpp文件中添加如下的代码:  



#include 〃test1。h〃  



int MsgBox(LPCTSTR lpText;  



LPCTSTR lpCaption;  



UINT uType)  



{  



return MessageBox(NULL;lpText;lpCaption;uType);  



}  



编译该工程,在Debug 目录下生成文件msgbox。lib和msgbox。dll。  



在 “13。2。2  链接应用程序到DLL”小节中将讲述如何使用在本节中所 

创建的DLL:msgbox。dll。  



     (2) 使用关键字__declspec(dllexport)  



从DLL中导出文件的另一种方法是在定义函数时使用__declspec 

(dllexport)关键字。这种方法不需要使用DEF文件。  



仍使用前面的例子,在工程中删除msgbox。def文件,将msgbox。h文件 

修改如下:  



#include   



extern 〃C〃 __declspec(dllexport) int MsgBox(  



// 消息框的文本  



LPCTSTR lpText=〃虽然这个例子有一些幼稚,但它工作得非常的好 !〃;  



// 消息框的标题  



LPCTSTR lpCaption=〃一个简单的例子〃;  



// 消息框的样式  



UINT uType=MB_OK);  



msgbox。cpp文件并不需要做任何修改,重新编译该工程,在Debug 目 


…………………………………………………………Page 669……………………………………………………………

录下仍生成两个文件msgbox。lib和msgbox。dll。  



在下一小节 “13。2。2  链接应用程序到DLL”中讲述了如何在应用程序 

中使用所创建的DLL:msgbox。dll的导出函数MsgBox()。  



使用__declspec(dllexport)从DLL中导出类的语法如下:  



class __declspec(dllexport) CDemoClass  



{  



。。。  



}  



    



  l 注意:  



  l 如果在使用__declspec(dllexport)的同时指定了调用协议关键 

    字,则必须将__declspec(dllexport)关键字放在调用协议关键字 

    的左边。如:  



  l int __declspec(dllexport) __cdacl MyFunc();  



  l 在32位版本的Visual C++中,__declspec(dllexport)和 

   __declspec(dllimport)代替了16版本中使用的__export关键字。 

    因此,在将16位的DLL源代码移植到Win32平台时,需要把每一处 

   __export替换为__declsped(dllexport)。  



如何从这两种导出函数的方法中作出选择,可以从下面的几个方面考 

虑:  



  l 如果需要使用导出顺序值 (export ordinal value),那么应该 

   使用DEF文件来导出函数。只在使用DEF文件导出函数才能指定导 

    出函数的顺序值。使用顺序值的一个好处是当向DLL中添加新的函 

    数时,只要新的导出函数的顺序值大于原有的导出函数,就没有 

    必要重新链接使用隐含链接的应用程序。相反,如果使

返回目录 上一页 下一页 回到顶部 0 0

你可能喜欢的