方式一: 采用 LoadLibraryEx
若DLL不在调用方的同一目录下,可以用LoadLibrary(L"DLL绝对路径")加载。但若调用的DLL内部又调用另外一个DLL,此时调用仍会失败。解决办法是用LoadLibraryEx:
LoadLibraryEx(“DLL绝对路径”, NULL, LOAD_WITH_ALTERED_SEARCH_PATH);
通过指定LOAD_WITH_ALTERED_SEARCH_PATH,让系统DLL搜索顺序从DLL所在目录开始。
方式二: 采用 SetCurrentDir
跨目录调用dll,你应该这样
- 用GetCurrentDir保存当前的工作目录
- 用SetCurrentDir将当前的工作目录,设置为你的DLL所在的路径,需要使用绝对路径
- 用LoadLibrary你的DLL
- 使用SetCurrentDir恢复到原来的工作路径
TCHAR chCurDir[MAX_PATH] = {0};
GetCurrentDirectory(MAX_PATH, chCurDir);
SetCurrentDirectory(_T("E:\\test\\"));
m_hDLL = LoadLibrary(_T("MyTest.dll"));
SetCurrentDirectory(chCurDir);
dll的加载顺序
- EXE所在目录;
- 当前目录GetCurrentDirectory();
- 系统目录GetSystemDirectory();
- WINDOWS目录GetWindowsDirectory();
- 环境变量 PATH 所包含的目录。
使用loadlibrary加载dll使用的路径时,这个函数会忽略这个路径,只会按既定规则加载dll。所以如果要加载指定目录的dll,可以用上述两个解决方案。
后续
最近又遇到一个126的问题, 采用上述两种方式依然无法解决
- 问题描述:
Windows 64位系统, 加载32位dll, DLL加载路径为 “C:\Windows\System32\xxx.dll” - 解决办法:
将上述加载失败的DLL, 复制一份到"C:\Windows\SysWOW64"下面;
程序不用更改, 路径参数依然填写"C:\Windows\System32\xxx.dll";
问题解决! - 原因:
简单来说可以从SysWOW64的全称找到原因:
32bit Windows On 64bit Windows(在64位Windows上的32位Windows)
详情可参考以下链接:
questions from stack overflow
什么是Wow64?