Как преобразовать 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)