关于动态链接库
动态链接允许一个模块仅需载入或执行一个导出的DLL函数。动态链接不同于我们熟悉的静态链接,使用静态链接时,链接程序会把库函数代码拷贝到每个需要的模块中。
动态链接的类型
调用一个DLL中的函数有两种方法:
载入时动态链接(load-time dynamic linking),模块非常明确调用某个导出函数,使得他们就像本地函数一样。这需要链接时链接那些函数所在DLL的导入库,导入库向系统提供了载入DLL时所需的信息及DLL函数定位。更多信息请参照“载入时动态链接”。
运行时动态链接(run-time dynamic linking),运行时可以通过LoadLibrary或LoadLibraryEx函数载入DLL。DLL载入后,模块可以通过调用GetProcAddress获取DLL函数的出口地址,然后就可以通过返回的函数指针调用DLL函数了。如此即可避免导入库文件了。更多信息,请参照“使用运行时动态链接库”。
DLL及内存管理
每个程序载入DLL后都回把它映射到虚地址空间,然后才能调用DLL的出口函数。
系统为每个DLL维护一个线程级的引用计数,一旦一个线程载入了该DLL,引用计数将会加1。而程序终止或者引用计数变为0(仅指运行时动态链接库),DLL就会释放占用程序的虚地址空间。
如同其他函数一样,一个DLL出口函数在线程的上下文中执行(或者说被调用),因此,以下情况同样适用:
调用DLL的程序线程可以使用通过DLL函数打开的句柄,相似的是,程序中任意线程打开的句柄也可以被DLL函数所使用。
DLL使用调用线程的栈空间及调用程序的虚地址空间。
DLL从调用程序的虚地址空间中分配内存。
动态链接的优点
相对于静态链接来说,动态链接具有以下优势:
多个程序把相同的DLL载入到相同的基地址,共享其在物理内存中的唯一拷贝。这样可以节省系统内存并减少交换。
当DLL中函数变更时,只要不是函数参数变更,调用方式改变或者返回值改变的话,调用它们的应用程序就不需要进行重新编译或重新链接。与之相反,静态链接对象代码就要求应用程序进行重新链接。
DLL可以提供(方便)售后服务。例如,修改显示驱动的DLL以支持程序装载时不支持的显示器。
不同编程语言编写的程序只要按照函数调用约定就可以调用同一个DLL函数。调用约定(如C、Pascal或者标准调用)控制着调用时参数入栈的次序,而不用管是函数或调用函数负责清栈,或者参数是否在寄存器中。更多信息,可以参照您的编译器的相关文档。
使用DLL的一个潜在缺陷是应用程序不是自完备的,它需要依赖另外的DLL模块也要存在。如果使用载入时动态链接,程序启动时发现DLL不存在,系统将终止程序并给出错误信息。而使用运行时动态链接,系统不会终止,但丢失的DLL中的出口函数同样不可用。