windows平台下,dll 的加载方式分为显式加载和隐式加载。
隐式加载需要在程序链接期间指定依赖库的名称、路径等信息,程序运行时由系统自动进行处理;
显式加载则是直接调用系统 API LoadLibrary/FreeLibrary 来加载或卸载 dll 模块。通常情况下,这两种方式可以处理绝大多数的应用需求。
但是在 dll 文件较多时,需要将 dll 分类管理,又不想修改系统环境变量时,我们可以采用”延迟加载“的方式来解决该问题。假设当前的应用的目录结构如下所示:
RootPath
├── Basic
│ ├── Basic1.dll
│ └── Basic2.dll
├── Client1.dll
├── Client2.dll
├── Components
│ ├── Cpnt1.dll
│ └── Cpnt2.dll
└── MainExE
MainExE 为应用主程序,依赖 Client1.dll 和 Client2.dll 两个库;
Client1.dll 和 Client2.dll 分别依赖 Components 目录下的 Cpnt1.dll 和 Cpnt2.dll;
而 Cpnt1.dll 和 Cpnt2.dll 则依赖 Basic 目录下的 Basic1.dll 和 Basic2.dll。
在程序启动时,系统会首先搜索程序运行时当前目录,找到 Client1.dll 和 Client2.dll。
但是如果没有配置环境变量, Client1.dll 和 Client2.dll 在运行时无法找到 Cpnt1.dll 和 Cpnt2.dll 这两个库,所以我们在编译 Client1.dll 和 Client2.dll 时,
需要在”链接器“-”延迟加载的DLL"这一项中分别添加 “Cpnt1.dll"和”Cpnt2.dll";
同时,在 Client 调用 Cpnt 的方法之前,在 Client 代码中手动调用 LoadLibrary 方法并传入 Cpnt 库的路径,此时就可以成功调用 Cpnt 中的方法。
Cpnt 调用 Basic 也是一样的道理,伪代码如下所示:
#include <windows.h> // Client1 调用 Cpnt1 void ClientInvokeCpnt() { HMODULE hModule = LoadLibrary("Components/Cpnt1.dll"); InvokeCpntFunction(); if (hModule) { FreeLibrary(hModule); } }
#include <windows.h> // Cpnt1 调用 Basic1 void CpntInvokeBasic() { HMODULE hModule = LoadLibrary("../Basic/Basic1.dll"); InvokeBasicFunction(); if (hModule) { FreeLibrary(hModule); } }
用这种方法则无需修改环境变量。
标签:Cpnt1,hModule,Windows,Client1,Client2,dll,隐式,加载 From: https://www.cnblogs.com/huowenjie/p/18229365