CreateProcessW 不尊重命令行

mar*_*rcp 5 c++ winapi c++builder c++98

我正在尝试在 dll 中实现 CreateProcessW,以便用户可以在单独的进程中启动应用程序。

对于初学者,我正在对代码中的命令进行硬编码,直到我弄清楚为止。

我有

STARTUPINFO si = {sizeof(STARTUPINFO), 0};
si.cb = sizeof(si);
PROCESS_INFORMATION pi = {0};
LPCTSTR AppName=L"c:\\utilities\\depends.exe";
LPTSTR Command = L"c:\\utilities\\tee.exe";
if (CreateProcessW(AppName, Command, 0, 0, 0, CREATE_DEFAULT_ERROR_MODE, 0, 0, &si, &pi)) {
        CloseHandle(pi.hThread);
        CloseHandle(pi.hProcess);
        return GX_RESULT_OK;
    } else {
        .. show error msg
    }
Run Code Online (Sandbox Code Playgroud)

这将启动 Depends,但不会打开 Tee.exe。没有错误,它只是忽略命令行参数。参数是正确的,我可以在运行提示下运行它,它工作正常。如果我将 AppName 留空并指定 Depends.exe 作为命令参数它也可以工作,但如果我指定

LPTSTR Command = L"c:\\utilities\\depends.exe c:\\utilities\\tee.exe";
Run Code Online (Sandbox Code Playgroud)

我收到错误 3:“系统找不到指定的路径”。

此外,通过指定 lpCurrentDirectory 参数,它同样会被忽略。

SHR*_*SHR 3

您必须在以下位置提供可执行文件command

  • Appname应包含可执行文件的完整路径
  • Command还应包含argv[0]

如果你想t.txt用记事本打开文件,你可以给出如下:

  • Appname = "c:/windows/notepad.exe";
  • command = "notepad c:/temp/t.txt";

您甚至不必提供真实的程序名称,即使是假名称也可以完成这项工作,因为它只是一个占位符。

像这样:command = "fake c:/temp/t.txt";

现在在记事本.exe中:

  • argv[0] = "notepad";
  • argv[1] = "c:/temp/t.txt";

请参阅这个完整的示例:

#include <Windows.h>
#include <iostream>

using namespace std;

int main(){
    STARTUPINFO si = {sizeof(STARTUPINFO), 0};
    si.cb = sizeof(si);
    PROCESS_INFORMATION pi = {0};
    LPTSTR AppName=L"C:/Windows/notepad.exe";
    wchar_t Command[] = L"notepad C:/Temp/t.txt"; 
    DWORD res = CreateProcess(AppName, Command, 0, 0, 0, CREATE_DEFAULT_ERROR_MODE, 0, 0, &si, &pi);
    if (res) {
        CloseHandle(pi.hThread);
        CloseHandle(pi.hProcess);
        return 0;
    } else {
        cerr<<"error..."<<GetLastError()<<endl;
    }; 
    return 0;
}
Run Code Online (Sandbox Code Playgroud)

  • @marcp Heck,如果你愿意的话,可以不同意我的观点,但你真的认为 MSDN 弄错了吗?简单来说:*此函数的 Unicode 版本 CreateProcessW 可以修改此字符串的内容。因此,该参数不能是指向只读内存的指针(例如 const 变量或文字字符串)。如果此参数是常量字符串,则该函数可能会导致访问冲突。* (2认同)