Как преобразовать std:: wstring в LPCTSTR в С++?
У меня есть значение ключа реестра Windows в формате wstring
. Теперь я хочу передать его этому коду (первый аргумент - путь к javaw.exe):
std::wstring somePath(L"....\\bin\\javaw.exe");
if (!CreateProcess("C:\\Program Files\\Java\\jre7\\bin\\javaw.exe", <--- here should be LPCTSTR, but I have a somePath in wstring format..
cmdline, // Command line.
NULL, // Process handle not inheritable.
NULL, // Thread handle not inheritable.
0, // Set handle inheritance to FALSE.
CREATE_NO_WINDOW, // ON VISTA/WIN7, THIS CREATES NO WINDOW
NULL, // Use parent environment block.
NULL, // Use parent starting directory.
&si, // Pointer to STARTUPINFO structure.
&pi)) // Pointer to PROCESS_INFORMATION structure.
{
printf("CreateProcess failed\n");
return 0;
}
Как я могу это сделать?
Ответы
Ответ 1
Просто используйте c_str
функцию std::w/string
.
Смотрите здесь:
http://www.cplusplus.com/reference/string/string/c_str/
std::wstring somePath(L"....\\bin\\javaw.exe");
if (!CreateProcess(somePath.c_str(),
cmdline, // Command line.
NULL, // Process handle not inheritable.
NULL, // Thread handle not inheritable.
0, // Set handle inheritance to FALSE.
CREATE_NO_WINDOW, // ON VISTA/WIN7, THIS CREATES NO WINDOW
NULL, // Use parent environment block.
NULL, // Use parent starting directory.
&si, // Pointer to STARTUPINFO structure.
&pi)) // Pointer to PROCESS_INFORMATION structure.
{
printf("CreateProcess failed\n");
return 0;
}
Ответ 2
LPCTSTR
- старая реликвия. Это гибридный typedef, который либо определяет char*
, если вы используете многобайтовые строки, либо wchar_t*
, если вы используете Unicode. В Visual Studio это можно изменить в общих настройках проекта в разделе "Набор символов".
Если вы используете Unicode, то:
std::wstring somePath(L"....\\bin\\javaw.exe");
LPCTSTR str = somePath.c_str(); // i.e. std::wstring to wchar_t*
Если вы используете многобайтовый, используйте этот помощник:
// wide char to multi byte:
std::string ws2s(const std::wstring& wstr)
{
int size_needed = WideCharToMultiByte(CP_ACP, 0, wstr.c_str(), int(wstr.length() + 1), 0, 0, 0, 0);
std::string strTo(size_needed, 0);
WideCharToMultiByte(CP_ACP, 0, wstr.c_str(), int(wstr.length() + 1), &strTo[0], size_needed, 0, 0);
return strTo;
}
то есть. std::wstring
до std::string
, который будет содержать многобайтную строку, а затем char*
:
LPCTSTR str = ws2s(somePath).c_str();
Ответ 3
Наконец, решил использовать CreateProcessW как палум, упомянутый с небольшими исправлениями - значения должны быть заброшены (в противном случае я получаю ошибку):
STARTUPINFOW si;
memset(&si, 0, sizeof (STARTUPINFOW));
si.cb = sizeof (STARTUPINFOW);
si.dwFlags = STARTF_USESHOWWINDOW;
si.wShowWindow = FALSE;
PROCESS_INFORMATION pi;
memset(&pi, 0, sizeof (PROCESS_INFORMATION));
std::wstring cmdline(L" -jar install.jar");
if (!CreateProcessW((LPCWSTR)strKeyValue.c_str(),
(LPWSTR)cmdline.c_str(), // Command line.
NULL, // Process handle not inheritable.
NULL, // Thread handle not inheritable.
0, // Set handle inheritance to FALSE.
CREATE_NO_WINDOW, // ON VISTA/WIN7, THIS CREATES NO WINDOW
NULL, // Use parent environment block.
NULL, // Use parent starting directory.
&si, // Pointer to STARTUPINFO structure.
&pi)) // Pointer to PROCESS_INFORMATION structure.
{
printf("CreateProcess failed\n");
return 0;
}
Ответ 4
Самый безопасный способ взаимодействия с классами stdlib с TCHAR
- использовать std::basic_string<TCHAR>
и окружать необработанные строки макросом TEXT()
(поскольку TCHAR
может быть узким и широким в зависимости от настроек проекта).
std::basic_string<TCHAR> somePath(TEXT("....\\bin\\javaw.exe"));
Так как вы не выиграете конкурсы на стили, которые делают это... другой правильный метод - явно использовать узкую или широкую версию функции WinAPI. Например. в этом конкретном случае:
- с
std::string
используйте CreateProcessA
(который использует LPCSTR
, который является typedef из char*
)
- с
std::u16string
или std::wstring
используйте CreateProcessW
(который использует LPCWSTR
, который является typedef из wchar_t*
, который является 16-разрядным в Windows)
В С++ 17 вы можете сделать:
std::filesystem::path app = "my/path/myprogram.exe";
std::string commandcall = app.filename.string() + " -myAwesomeParams";
// define si, pi
CreateProcessA(
const_cast<LPCSTR>(app.string().c_str()),
const_cast<LPSTR>(commandcall.c_str()),
nullptr, nullptr, false, CREATE_DEFAULT_ERROR_MODE, nullptr, nullptr,
&si, &pi)