C++ CreateProcess+;调用命令行工具

C++ CreateProcess+;调用命令行工具,c++,windows,C++,Windows,我对线程和进程非常陌生,我发现我无法让CreateProcess()启动一个可执行文件,尽管论坛上还有很多关于它的帖子。从我对其工作原理的微薄理解来看,我认为我设置了正确的参数,但我得到了一个创建过程失败(267)错误。我试图运行的可执行文件是一个命令行工具,它属于名为xst的Xilinx套件。我想要的是在由path全局变量定义的目录中运行xst,这样它就可以处理存储在其中的一些文件。CreateProcess()的参数是否有误 #包括“stdafx.h” #包括 #包括 #包括 #包括 #包括

我对线程和进程非常陌生,我发现我无法让CreateProcess()启动一个可执行文件,尽管论坛上还有很多关于它的帖子。从我对其工作原理的微薄理解来看,我认为我设置了正确的参数,但我得到了一个
创建过程失败(267)
错误。我试图运行的可执行文件是一个命令行工具,它属于名为xst的Xilinx套件。我想要的是在由
path
全局变量定义的目录中运行xst,这样它就可以处理存储在其中的一些文件。CreateProcess()的参数是否有误

#包括“stdafx.h”
#包括
#包括
#包括
#包括
#包括
#包括
#包括
#包括
std::string path=“C:\\FPGA\\BSP\\BSP\\Xilinx\\SingleItemTest\\”;
void testXST(std::string文件路径,std::string参数){
STARTUPINFO si;
处理信息;
零内存(&si,sizeof(si));
si.cb=sizeof(si);
零内存(&pi,sizeof(pi));
//启动子进程。
如果(!CreateProcess(
LPTSTR(filePath.c_str()),
LPTSTR(arguements.c_str()),
NULL,//进程句柄不可继承
NULL,//线程句柄不可继承
FALSE,//将句柄继承设置为FALSE
0,//没有创建标志
NULL,//使用父级的环境块
LPTSTR(path.c_str()),
&si,//指向STARTUPINFO结构的指针
&pi)//指向进程信息结构的指针
)
{
printf(“CreateProcess失败(%d)。\n”,GetLastError());
返回;
}
//等待子进程退出。
WaitForSingleObject(pi.hProcess,无限);
//关闭进程和线程句柄。
CloseHandle(pi.hProcess);
CloseHandle(pi.hThread);
}
int main()
{
std::string xstPath=“C:\\Xilinx\\14.7\\ISE\u DS\\ISE\\bin\\nt\\xst.exe”;
std::string args=“-h”;
testXST(xstPath,args);
返回0;
}

我为xst设置了环境变量,因此我可以在命令行的任何位置调用它,但由于我给出了不重要的可执行文件的直接路径,对吗?

这里有几个问题

首先,@PaulMcKenzie在评论中指出,
lpCommandLine
参数的类型是
LPTSTR
,而不是
LPCTSTR
。因此,您不能传递std::string::c_str()的返回值,因为它返回常量缓冲区。您需要将
std::string
的内容复制到
char
数组中,并将其传递到
CreateProcess()


其次,您没有为
lpCommandLine
参数传递正确的值。它的值不是进程的参数,而是要执行的整个命令行。除任何参数外,还必须包括可执行文件的路径。可以将此视为在命令提示符下键入以运行相同的内容。

这里有几个问题

首先,@PaulMcKenzie在评论中指出,
lpCommandLine
参数的类型是
LPTSTR
,而不是
LPCTSTR
。因此,您不能传递std::string::c_str()的返回值,因为它返回常量缓冲区。您需要将
std::string
的内容复制到
char
数组中,并将其传递到
CreateProcess()


其次,您没有为
lpCommandLine
参数传递正确的值。它的值不是进程的参数,而是要执行的整个命令行。除任何参数外,还必须包括可执行文件的路径。可以将此视为在命令提示符下键入以运行相同的内容。

LPTSTR(filePath.c_str()),
--为什么要强制转换
(LPTSTR)
??如果您正在构建UNICODE应用程序,这将不起作用。删除强制转换,如果存在编译器错误,请使用适当的字符串类型修复错误,不要通过强制转换来掩盖错误。您需要
std::wstring
。并在字符串文本前面加上
L
。然后删除所有强制转换。另外,如果您使用的是Unicode版本,请小心选择
CreateProcess
。第二个参数必须是可写的TCHAR缓冲区,而不是字符串文字。
LPTSTR(filePath.c_str()),
--为什么要强制转换
(LPTSTR)
??如果您正在构建UNICODE应用程序,这将不起作用。删除强制转换,如果存在编译器错误,请使用适当的字符串类型修复错误,不要通过强制转换来掩盖错误。您需要
std::wstring
。并在字符串文本前面加上
L
。然后删除所有强制转换。另外,如果您使用的是Unicode版本,请小心选择
CreateProcess
。第二个参数必须是可写的TCHAR缓冲区,而不是字符串文字。好的,这很有意义。我希望exe在特定目录中运行,以便它可以访问某些文件。我把它放在lpCurrentDirectory参数中正确吗?@nhoughto是的,正确。
lpCurrentDirectory
的值将是新进程的初始工作文件夹。确定,这很有意义。我希望exe在特定目录中运行,以便它可以访问某些文件。我把它放在lpCurrentDirectory参数中正确吗?@nhoughto是的,正确。
lpCurrentDirectory
的值将是新进程的初始工作文件夹。
#include "stdafx.h"
#include <stdio.h>     
#include <stdlib.h>     
#include <iostream>
#include <fstream>
#include <sddl.h>
#include <windows.h>
#include <AccCtrl.h>
#include <Aclapi.h>

std::string path = "C:\\FPGA\\BSP\\BSP\\Xilinx\\SingleItemTest\\";

void testXST(std::string filePath, std::string arguements) {
    STARTUPINFO si;
    PROCESS_INFORMATION pi;

    ZeroMemory(&si, sizeof(si));
    si.cb = sizeof(si);
    ZeroMemory(&pi, sizeof(pi));

    // Start the child process. 
    if (!CreateProcess(
        LPTSTR(filePath.c_str()),   
        LPTSTR(arguements.c_str()),         
        NULL,           // Process handle not inheritable
        NULL,           // Thread handle not inheritable
        FALSE,          // Set handle inheritance to FALSE
        0,              // No creation flags
        NULL,           // Use parent's environment block
        LPTSTR(path.c_str()),            
        &si,            // Pointer to STARTUPINFO structure
        &pi)           // Pointer to PROCESS_INFORMATION structure
        )
    {
        printf("CreateProcess failed (%d).\n", GetLastError());
        return;
    }

    // Wait until child process exits.
    WaitForSingleObject(pi.hProcess, INFINITE);

    // Close process and thread handles. 
    CloseHandle(pi.hProcess);
    CloseHandle(pi.hThread);
}

int main()
{
    std::string xstPath = "C:\\Xilinx\\14.7\\ISE_DS\\ISE\\bin\\nt\\xst.exe";
    std::string args = " -h";
    testXST(xstPath, args);
    return 0;
}