如何将程序的执行文件和静态加载动态库放在不同的目录

来源:互联网 发布:telnet 端口打开 编辑:程序博客网 时间:2024/05/16 15:02

一般windows程序的exe和dll需要放在同一个目录,静态加载才不会报错,否则需要修改path环境变量,将所有没有和exe放在同一目录的dll的路径加在path环境变量中。

有没有一种方法不去手动修改path环境变量并且可以将exe和dll随心所欲的改变路径呢?我没有发现,但是我们可以将修改环境变量这件事情交给我们的程序本身来处理,那么从现象上来看就是我们不需要修改环境变量而可以将dll从exe目录中拿走,放到你所希望的位置。

其实实现这个想法不难,反而很简单。

假设我们的exe模块叫“A.exe”,依赖“B.dll”和“C.dll",你希望将“B.dll”放在”./B“目录下,而把"C.dll"放在”./C"目录下。我们的处理思路是,再写一个壳程序,假如说叫“E.exe”,“E.exe”除了系统的库之外不依赖任何自己开发或者第三方的库,那么理论上它放在哪里都是可以启动的。

在“E.exe”的代码中,我们设置环境变量,将“B.dll”和“C.dll"所在的路径加到Path环境变量中,并启动“A.exe”,那么在启动“A.exe”时就能自然而然加载到所依赖的库了。在启动“A.exe”完成之后将Path环境变量还原。所以在启动“A.exe”前后,Path环境变量看似并没有改变,但是我们棘手的问题却解决了。

当然为了避免用户手动去点击“A.exe”弹出错误窗口的不好的用户体验,可以将“A.exe”编译成dll,提供一个能够启动程序功能的导出API,在“E.exe”中动态加载“A.dll”并调用导出API,达到启动程序的目的。

但是exe重编程dll可能引起的问题有很多,比如MFC中就会存在很多的坑,那么还有一种简单的思路,那就是直接改名,将编译完成的“A.exe”直接改名成“A.dll”,然后在“E.exe”中通过CreateProcess的方式启动“A.dll”,然后就没“E.exe”什么事,可以退出歇着去了。

下面是我自己使用的“E.exe”的实现代码,相当简单,值得注意的是,“E.exe”运行时最好不要出现窗口,不然会很难看,至于怎么让程序不出现窗口,应该网上可以找到很多教程。

//E.exe Main.cppclass ChangePath{public:ChangePath(){size_t len = 0;char sz[2048] = {};getenv_s(&len, sz,2048,"PATH");m_szPath = sz;std::string szAddPath = "";//你的dll所在的绝对路径,使用“;”隔开std::string szNewPath = m_szPath + ";";szNewPath += szAddPath;_putenv_s("PATH",szNewPath.c_str());}~ChangePath(){_putenv_s("PATH",m_szPath.c_str());}private:std::string m_szPath;};ChangePath changepath;//RAII修改Path环境变量void LoadInstance(void *param){#if 1//假dllUSES_CONVERSION;char apppath[1024] = {};std::string exename = "C:/A.dll";strcpy_s(apppath,exename.c_str());char commandline[2048] = {};strcat_s(commandline,_countof(commandline),apppath);LPWSTR pszCmdLinew = GetCommandLineW();int argc = 0;CString FilePath = _T("");LPWSTR *argv = CommandLineToArgvW(pszCmdLinew, &argc);if (argv != NULL){for (int i = 1; i < argc; ++i){strcat_s(commandline,_countof(commandline)," ");strcat_s(commandline,_countof(commandline),W2A(argv[i]));};LocalFree(argv);}PROCESS_INFORMATION pi;STARTUPINFOA si = {sizeof(si)};CreateProcessA(apppath,commandline,NULL,NULL, FALSE, 0, NULL, NULL, &si, &pi);#else//真dlltypedef void (* InvokFunc)();//定义函数指针类型HINSTANCE hInst;hInst=LoadLibrary(_T("A.dll"));//动态加载Dllint error = GetLastError();InvokFunc invokFunc=(InvokFunc)GetProcAddress(hInst,"Entrance");//获取Dll的导出函数error = GetLastError();if(invokFunc){invokFunc();}::FreeLibrary(hInst);//释放Dll函数#endif}int main(){LoadInstance(NULL);return 1;}


0 0
原创粉丝点击