动态链接库与静态链接库

来源:互联网 发布:php发短信 编辑:程序博客网 时间:2024/04/27 23:32

    关于动态链接库

 

  动态链接允许一个模块仅需载入或执行一个导出的DLL函数。动态链接不同于我们熟悉的静态链接,使用静态链接时,链接程序会把库函数代码拷贝到每个需要的模块中。

动态链接的类型

  调用一个DLL中的函数有两种方法:

  • 载入时动态链接(load-time dynamic linking),模块非常明确调用某个导出函数,使得他们就像本地函数一样。这需要链接时链接那些函数所在DLL的导入库,导入库向系统提供了载入DLL时所需的信息及DLL函数定位。更多信息请参照“载入时动态链接”。(相当于注册,当作API函数来使用,其实API函数就存放在系统DLL库当中
  • 运行时动态链接(run-time dynamic linking),运行时可以通过LoadLibrary或LoadLibraryEx函数载入DLL。DLL载入后,模块可以通过调用GetProcAddress获取DLL函数的出口地址,然后就可以通过返回的函数指针调用DLL函数了。如此即可避免导入库文件了。更多信息,请参照“使用运行时动态链接库”。(即使用函数loadlibrary来表明库文件的存在供运行时载入,在编译和链接的时候不会检查库文件(甚至不会检查它是否存在),要到程序运行的时候才加载该文件,此时可能会出现载入库文件错误

 

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中的出口函数同样不可用。

 

      静态链接库的优点:1.代码装载速度快,执行速度略比动态库快。但装载完成之后,二者速度差不多。2.虽然使用lib文件程序文件较大,但一个EXE文件就能使程序运行起来。