我想我的问题是这样的 - 如何将exe位置的目录作为LPCWSTR获取,以便我可以将其输入到我的代码中
#include <iostream>
#include <Windows.h>
int main(int argc, char **argv)
{
WIN32_FIND_DATA a;
HANDLE swap = FindFirstFile(/*(LPCWSTR)__exe_directory__*/,&a);
if (swap!=INVALID_HANDLE_VALUE)
{
do
{
char *sptn = new char [lstrlen(a.cFileName)+1];
for (int c=0;c<lstrlen(a.cFileName);c++)
{
sptn[c]=char(a.cFileName[c]);
}
sptn[lstrlen(a.cFileName)]='\0';
std::cout<<sptn<<std::endl;
}
while (FindNextFile(swap,&a));
}
else std::cout<<"undetected file\n";
FindClose(swap);
system("pause");
}
它将返回目录中列出的文件而不会出错。我知道我的代码已经在给定目录的情况下工作了,我已经测试了它。
答案 0 :(得分:5)
关键是使用GetModuleFileName()
(传递nullptr
作为模块句柄,引用当前进程EXE),然后调用PathRemoveFileSpec()
(或PathCchRemoveFileSpec()
,如果你不要关心Windows 8之前的Windows版本,以便从路径中删除文件规范。
要使用PathRemoveFileSpec()
,您必须与MSDN documentation中所述的Shlwapi.lib相关联。
请参阅此可编辑代码作为示例:
#include <iostream> // For console output
#include <exception> // For std::exception
#include <stdexcept> // For std::runtime_error
#include <string> // For std::wstring
#include <Windows.h> // For Win32 SDK
#include <Shlwapi.h> // For PathRemoveFileSpec()
#pragma comment(lib, "Shlwapi.lib")
// Represents an error in a call to a Win32 API.
class win32_error : public std::runtime_error
{
public:
win32_error(const char * msg, DWORD error)
: std::runtime_error(msg)
, _error(error)
{ }
DWORD error() const
{
return _error;
}
private:
DWORD _error;
};
// Returns the path without the filename for current process EXE.
std::wstring GetPathOfExe()
{
// Get filename with full path for current process EXE
wchar_t filename[MAX_PATH];
DWORD result = ::GetModuleFileName(
nullptr, // retrieve path of current process .EXE
filename,
_countof(filename)
);
if (result == 0)
{
// Error
const DWORD error = ::GetLastError();
throw win32_error("Error in getting module filename.",
error);
}
// Remove the file spec from the full path
::PathRemoveFileSpec(filename);
return filename;
}
int main()
{
try
{
std::wcout << "Path for current EXE:\n"
<< GetPathOfExe()
<< std::endl;
}
catch (const win32_error & e)
{
std::cerr << "\n*** ERROR: " << e.what()
<< " (error code: " << e.error() << ")"
<< std::endl;
}
catch (const std::exception& e)
{
std::cerr << "\n*** ERROR: " << e.what() << std::endl;
}
}
在控制台中:
C:\Temp\CppTests>cl /EHsc /W4 /nologo /DUNICODE /D_UNICODE get_exe_path.cpp get_exe_path.cpp C:\Temp\CppTests>get_exe_path.exe Path for current EXE: C:\Temp\CppTests
<强> PS 强>
在您的代码中,您似乎引用了FindFirtFile()
的Unicode版本(即FindFirstFileW()
,因为在评论中您期望LPCWSTR
,即const wchar_t*
),但是以下代码使用ANSI / MBCS字符串(即char*
)。
我建议您始终在现代Windows C ++代码中使用 Unicode UTF-16 wchar_t*
字符串:它更适合国际化,仅适用于现代Win32 API附带Unicode版本。
另请注意,由于您使用的是C ++,因此最好使用强大的方便的字符串类(例如std::wstring
用于Microsoft Visual C ++的Unicode UTF-16字符串),而不是类似C的原始字符指针。使用API接口上的原始指针(因为Win32 API具有C接口),然后安全地转换为std::wstring
。
答案 1 :(得分:3)
在UNICODE构建中使用GetModuleFileName函数以宽字符串形式获取可执行文件的完整文件名。
然后,搜索最后一个&#39; \&#39;字符并将其替换为0。
完成。
#include <Windows.h>
#include <stdio.h>
int main( void ) {
wchar_t szExeFullPath[ MAX_PATH ];
if ( GetModuleFileName( NULL, szExeFullPath, _countof( szExeFullPath ) ) ) {
wchar_t * pszLastAntiSlash = wcsrchr( szExeFullPath, L'\\' );
if ( pszLastAntiSlash ) {
*pszLastAntiSlash = 0;
wprintf( L"Exe full path is %s\n", szExeFullPath );
}
}
return 0;
}