GetCurrentDirectory、SetCurrentDirectory和GetModuleFileName

来源:互联网 发布:java并行框架 编辑:程序博客网 时间:2024/06/06 23:50

一.学习 GetModuleFileName:

方法一:

想要访问执行程序(.exe)路径下的文件,有以下几个步骤:

    1.先通过函数GetModuleFileName获取执行程序的绝对路径

    TCHAR szPath[ MAX_PATH ] = {0};
    GetModuleFileName( NULL, szPath, MAX_PATH );

    其中GetModuleFileName函数为windows的API函数,使用的时候需要包含windows.h的头文件;MAX_PATH是一个宏定义,值为260。执行完GetModuleFileName函数之后,szPath数组中保存的就是执行程序当前的绝对路径。

假设执行程序xp.exe的绝对路径为C:\Program Files\Dll\xp.exe,那么szPath数组中存储的值就是C:\Program Files\Dll\xp.exe。

    2.根据执行程序的绝对路径得到需要访问的文件的绝对路径。

    这里需要用到_tcsrchr函数来将获取到的执行程序的绝对路径中的执行程序的名称去掉。即将上面路径C:\Program Files\Dll\xp.exe中的xp.exe去掉,只得到C:\Program Files\Dll\。方法如下:

    (_tcsrchr(szPath,_T('\\')))[1] = 0;

    注:_tcsrchr包含在头文件tchar.h中。

    3.将要访问的文件的名称添加到已经得到的路径中。

    假设需要访问的文件是1.txt,那么如下:

     CString strPath = szPath;
     strPath = strPath + "1.txt";

     在非MFC中,CString包含在atlstr.h头文件中,在MFC中包含在cstring.h的头文件中。

     程序整体如下:

     TCHAR szPath[ MAX_PATH ] = {0};
     if ( GetModuleFileName( NULL, szPath,MAX_PATH ) )
     {
      (_tcsrchr(szPath,_T('\\')))[1] = 0;
     }

     CString strPath = szPath;
     strPath = strPath + "1.txt";

 

函数原型分析:

GetModuleFileName函数原型
DWORD GetModuleFileName(
   HMODULE hModule,     // handle to module。将要得到的模块的句柄。如果是当前模块,NULL
   LPTSTR lpFilename,   // path buffer   得到的文件名。
   DWORD nSize          // size of buffer   一般MAX_PATH就可以了
);
可以通过以下方法获取路径名
 
char szCurPath[MAX_PATH];//定义一个字符串数据,长度为MAX_PATH ,MAX_PATH是预定义的宏,一般是个数字常量如256
 
GetModuleFileName(NULL,szCurPath,_MAX_PATH);
 
char* p = szCurPath;
 
while(strchr(p,'\\'))
{
         p = strchr(p,'\\');
         p++;
}
 
*p = '\0';
 
CString strSwfname = szCurPath; 
szCurPath就是去除文件名的路径名

 

 

二.GetCurrentDirectory

是的到应用程序的当前目录,但当前目录不一定等于应用程序执行文件的所在目录,一个应用程序被启动时,当前目录是可以被任意设置的。

可以理解成应用程序的当前工作目录。

原文标题:loadlibrary加载三方dll失败问题。 

前几天,在联网测试三方的dll,但是出现dll放到exe目录下面可以正常加载,如果单独放一个目录却出现126错误代码,找不到指定的模块。 由于三方dll使用zlib版本和exe使用的zlib版本不同,所以三方的dll又不能放到exe目录中,必须单独新建目录。于是在网上搜索得到以下解决方案:

方式一 采用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,你应该这样

1 用GetCurrentDir保存当前的工作目录
2 用SetCurrentDir将当前的工作目录,设置为你的DLL所在的路径,需要使用绝对路径
3 用LoadLibrary你的DLL
4 使用SetCurrentDir恢复到原来的工作路径

 

[cpp] view plain copy
  1. TCHAR chCurDir[MAX_PATH] = {0};  
  2. GetCurrentDirectory(MAX_PATH, chCurDir);  
  3. SetCurrentDirectory(_T("E:\\test\\"));  
  4. m_hDLL = LoadLibrary(_T("MyTest.dll"));  
  5. SetCurrentDirectory(chCurDir);  


 

总结:dll的加载顺序:

(1)EXE所在目录;
(2)当前目录GetCurrentDirectory();
(3)系统目录GetSystemDirectory();
(4)WINDOWS目录GetWindowsDirectory();
(5)环境变量 PATH 所包含的目录。

所以使用loadlibrary加载dll使用的路径,但是这个函数会忽略这个路径,只会按既定规则加载dll。所以如果要加载指定目录的dll,可以用上述两个解决方案。

转自:http://blog.csdn.net/lijianli9/article/details/9091453

ps:我用的是GetModuleFileName获取exe路径,然后再掉LoadLibraryEx解决LoadLibrary返回126的错误

 

 

 

阅读全文
0 0
原创粉丝点击