DLL动态库获取自身所在路径的两种方法

方法一:通过动态库名称获取加载动态库的路径

#include<string>
#include<Windows.h>
#include<iostream>
void PrintDllPath()
{
    HMODULE hDllHandle = GetModuleHandleA("dlltest1.dll");//”dlltest1.dll“是本代码片所属动态库名称
    char cDllPathBuff[MAX_PATH];
    memset(cDllPathBuff, 0, MAX_PATH);
    auto getdllpathstatus = GetModuleFileNameA((HMODULE)hDllHandle, cDllPathBuff,
                            MAX_PATH);
    std::string strDllPathBuff = cDllPathBuff;
    auto nPathTruncLoc = strDllPathBuff.find_last_of('\\');
    strDllPathBuff = strDllPathBuff.substr(0, nPathTruncLoc + 1);//strDllPathBuff 为本动态库所在路径
    std::cout << strDllPathBuff << std::endl;
}

注意:
该方法在执行的时候是根据动态库名称,去调用动态库的可执行文件所在路径及其相对路径去搜索,因此当搜索路径下有多个同名的动态库,该方法不能准确获取到所加载动态库的路径。
例如在可执行文件目录下有多个文件夹中存有同名的动态库,则会找到第一个搜索的含有该同名动态库的路径,可能与加载的动态库不是同一个路径。

针对上述问题,可通过方法二解决。

方法二 通过动态库导出函数的内存地址获取当前加载动态库所在路径

#include<string>
#include<Windows.h>
#include<iostream>
void PrintDllPath()
{
    HMODULE hCaller = NULL;
    char szModuleName[MAX_PATH];
    void* pRetAddr = &PrintDllPath;//取本导出函数的地址
    GetModuleHandleExA(GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS, (LPCSTR)pRetAddr, &hCaller);
    GetModuleFileNameA(hCaller, szModuleName, _countof(szModuleName));
    std::string strDllPathBuff = szModuleName;
    auto nPathTruncLoc = strDllPathBuff.find_last_of('\\');
    strDllPathBuff = strDllPathBuff.substr(0, nPathTruncLoc + 1);
    std::cout << strDllPathBuff << std::endl;
}

该方法通过运行时函数在内存的地址找到当前可执行程序所加载的动态库路径,避免了方法一中多同名动态库混淆,导致获取路径不正确的情况。

注意:
如果需要在类成员函数里面获取路径,则可将对应成员函数声明为静态函数,再通过该函数地址用方法二获取动态库所在路径

//TestDll.h头文件
class __declspec(dllexport) TestDll: public TopCls
{
public:
    void PrintDllPath();
private:
    static std::string GetDllPath();//获取当前动态库所在路径
};
void TestDll::PrintDllPath()
{
    cout << GetDllPath() << endl;
}

std::string TestDll::GetDllPath()
{
    HMODULE hCaller = NULL;
    char szModuleName[MAX_PATH];
    void* pRetAddr = &GetDllPath;//静态成员函数
    GetModuleHandleExA(GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS, (LPCSTR)pRetAddr, &hCaller);
    GetModuleFileNameA(hCaller, szModuleName, _countof(szModuleName));
    std::string strDllPathBuff = szModuleName;
    auto nPathTruncLoc = strDllPathBuff.find_last_of('\\');
    strDllPathBuff = strDllPathBuff.substr(0, nPathTruncLoc + 1);
    return strDllPathBuff ;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

懒人空想家

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值