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