zoukankan      html  css  js  c++  java
  • 动态库的装载与卸载

    相关的三个函数:

    LoadLibrary,GetProcAddress,FreeLibrary

    动态载入 DLL
    动态载入方式是指在编译之前并不知道将会调用哪些 DLL 函数, 完全是在运行过程中根据需要决定应调用哪些函数。
    方法是:用 LoadLibrary 函数加载动态链接库到内存,用 GetProcAddress函数动态获得 DLL 函数的入口地址。当一个 DLL 文件用 LoadLibrary 显式加载后,在任何时刻均可以通过调用 FreeLibrary 函数显式地从内存中把它给卸载。

    个人觉得,安全稳妥起见,装载过程应放在构造函数中,而释放动态库应在析构函数中进行操作。这样可以防止多次加载后,而在释放中少释放导致内存浪费问题。

     

    (1)LoadLibrary 函数
    注:Delphi 中还提供了 SafeLoadLibrary 函数,它封装了 Loadlibrary 函数,可以装载由 Filename 参数指定的 WindowsDLL或 Linux 共享对象。它简化了DLL的装载并且使装载更加安全。
    [格式]:

    1. function LoadLibrary(LibFileName : PChar): Thandle;

    复制代码

    [功能]:加载由参数 LibFileName 指定的 DLL 文件。
    [说明]:参数 LibFileName 指定了要装载的 DLL 文件名,如果 LibFileName 没有包含一个路径,系统将按照:当前目录、Windows 目录、Windows 系统目录、包含当前任务可执行文件的目录、列在 PATH 环境变量中的目录等顺序查找文件。
    如果函数操作成功,将返回装载 DLL 库模块的实例句柄,否则,将返回一个错误代码,错误代码的定义如下表所示。

    错误代码

      含义

      0

      系统内存不够,可执行文件被破坏或调用非法

      2

      文件没有被发现

      3

      路径没有被发现

      5

      企图动态链接一个任务错误或者有一个共享或网络保护错误

      6

      库需要为每个任务建立分离的数据段  

      8

      没有足够的内存启动应用程序  

      10

      Windows  版本不正确  

      11

      可执行文件非法或不是Windows  应用程序,或在.  EXE映像中有错误  

      12

      应用程序为一个不同的操作系统设计(如  OS/2)  

      13

      应用程序为  MS  DOS   4. 0  设计  

      14

      可执行文件的类型不知道  

      15

      试图装载一个实模式应用程序(为早期Windows  版本设计)

      16

      试图装载包含可写的多个数据段的可执行文件的第二个实例  

      19

      试图装载一个压缩的可执行文件(文件必须被解压后才能被装载)  

      20

      DLL  文件非法

      21

      应用程序需要  32  位扩展

    假如在应用程序中用 LoadLibrary 函数装入某一个 DLL 前, 其他应用程序已把该 DLL 装入内存中了,则系统将不再装入该 DLL 的另一个实例,而是使该 DLL 的“引用计数”加 1 。
    (2)GetProcAddress 函数
    [格式]:

    1. function GetProcAddress(Module:Thandle; ProcName:PChar): TfarProc;

    复制代码

    [功能]:返回参数 Module 指定的模块中,由参数 ProcName 指定的过程或函数的入口地址。
    [说明]:参数 Module 包含被调用函数的 DLL 句柄,这个值由 LoadLibrary 返回, ProcName
    是指向含有函数名的以 nil 结尾的字符串指针,或者可以是函数的次序值,但大多数情况下,用函数名是一种更稳妥的选择。如果该函数执行成功,则返回 DLL 中由参数 ProcName 指定的过程或函数的入口地址,否则返回 nil 。
    (3)FreeLibrary 函数
    [格式]:

    1. procedure  FreeLibrary(Module: Thandle);

    复制代码

    [说明]:将由参数 Module 指定的 DLL 文件从内存中卸载 1 次。
    [说明]:Module 为 DLL 库的句柄。这个值由 LoadLibrary 返回。由于 DLL 在内存中只装载一次,因此调用 FreeLibrary 首先使 DLL 的引用计数减 1,如果计数减为 0 则卸载该 DLL。
    [注意]:每调用一次 LoadLibrary 函数就应调用一次 FreeLibrary 函数,以保证不会有多余的库模块在应用程序结束后仍留在内存中,否则导致内存泄漏。

    生活的残酷,让我们习惯了忘记疲倦,一直奔向远方,追寻着自己的梦想。
  • 相关阅读:
    内核初始化. Part 4【转】
    fixmap addresses原理【转】
    linux内核调试项【转】
    Linux Suspend流程分析【转】
    Linux电源管理-Suspend/Resume流程【转】
    Linux驱动开发常用调试工具---之内存读写工具devmem和devkmem【转】
    内核regmap机制【转】
    ARM NVIC控制器(基于cortex-M4)【转】
    Linux设备树语法详解-中断【转】
    [Go] 第一个单词首字母变大写:Ucfirst(),第一个单词首字母变小写:Lcfirst()
  • 原文地址:https://www.cnblogs.com/L-Arikes/p/4374715.html
Copyright © 2011-2022 走看看