首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

使用ShellExecuteEx并捕获标准输入/输出/错误

作为一个云计算领域的专家,我可以告诉你,使用ShellExecuteEx并捕获标准输入/输出/错误是一种在Windows操作系统中执行外部程序的方法。ShellExecuteEx是一个Windows API函数,它允许你以异步方式执行外部程序,并且可以捕获该程序的标准输入/输出/错误流。

在使用ShellExecuteEx时,你需要提供一个完整的命令行,包括可执行文件的路径和参数。你还可以指定一个STARTUPINFO结构体,该结构体包含了一些与新进程相关的信息,例如新进程的工作目录和环境变量。ShellExecuteEx函数将返回一个PROCESS_INFORMATION结构体,该结构体包含了新进程的一些信息,例如进程ID和主线程ID。

要捕获标准输入/输出/错误流,你需要使用CreatePipe函数创建一个管道,并将其与新进程的标准输入/输出/错误流关联。你可以使用ReadFile和WriteFile函数读取和写入管道中的数据。

以下是使用ShellExecuteEx并捕获标准输入/输出/错误流的一个示例代码:

代码语言:c++
复制
#include<Windows.h>
#include<iostream>

int main()
{
    // 创建管道
    HANDLE hReadPipe, hWritePipe;
    SECURITY_ATTRIBUTES sa = {sizeof(sa), NULL, TRUE};
    CreatePipe(&hReadPipe, &hWritePipe, &sa, 0);

    // 设置STARTUPINFO结构体
    STARTUPINFO si = {sizeof(si)};
    si.dwFlags = STARTF_USESHOWWINDOW | STARTF_USESTDHANDLES;
    si.hStdInput = hReadPipe;
    si.hStdOutput = hWritePipe;
    si.hStdError = hWritePipe;
    si.wShowWindow = SW_HIDE;

    // 调用ShellExecuteEx函数
    PROCESS_INFORMATION pi;
    ZeroMemory(&pi, sizeof(pi));
    ShellExecuteEx(&pi, NULL, "cmd.exe", "/c dir", NULL, SW_HIDE, &si);

    // 读取输出数据
    char buffer[4096];
    DWORD bytesRead;
    while (ReadFile(hReadPipe, buffer, sizeof(buffer), &bytesRead, NULL) && bytesRead > 0)
    {
        std::cout.write(buffer, bytesRead);
    }

    // 等待进程结束
    WaitForSingleObject(pi.hProcess, INFINITE);

    // 关闭句柄
    CloseHandle(pi.hProcess);
    CloseHandle(pi.hThread);
    CloseHandle(hReadPipe);
    CloseHandle(hWritePipe);

    return 0;
}

在这个示例中,我们使用ShellExecuteEx函数启动一个命令提示符窗口,并执行“dir”命令。我们将标准输入/输出/错误流都重定向到一个管道中,并使用ReadFile函数读取管道中的数据。最后,我们等待进程结束,并关闭所有句柄。

需要注意的是,使用ShellExecuteEx函数启动外部程序可能存在安全风险,因此应该谨慎使用。此外,如果你需要执行的外部程序是一个GUI程序,那么你需要使用其他方法来启动它,例如使用CreateProcess函数。

页面内容是否对你有帮助?
有帮助
没帮助

相关·内容

  • 第9章 Java高级编程

    第9章 Java高级编程   学习导读   本章将介绍Java语言中的一些高级特性:异常处理、多线程机制、流式IO以及网络通信,其中每个部分都能扩展成一个专题。读者在本章学习到的只是其中很小的一部分,但是能够从中了解一些基本概念和基本操作,为今后的深入学习打下基础。异常处理提供对错误的捕捉和处理机制;多线程机制使得程序的多个子任务能够“同时”执行;流式IO提供了对输入输出的读写机制;网络通信特性允许Java程序通过Socket实现底层通信,并利用Java提供的扩展组件实现高层服务。 教学重点与难点:   ●异常处理、多线程机制、流式IO以及网络通信的基本概念和模型   ●异常的捕获和处理   ●多线程的实现及Runnable接口的应用   ●流式输入输出的操作、文件读写及随机访问   ●基于Socket的面向连接的网络底层通信及高层服务 9.1 异常处理   在Java中,提供了错误捕捉和处理机制,即异常处理机制。在程序运行过程中发生错误时,Java允许其不按照正常路径完成任务,由发现错误的方法抛出封装了错误信息的对象(异常)到其调用程序,发出已经发生问题的信号,然后立即退出;而且,程序并不在调用该方法的代码处继续执行,而是由异常处理机制开始搜索一个能够处理这种特定错误情况的异常处理器。   异常(Exception)也称例外、差错、违例等,是特殊的运行错误对象,它对应于java中的特定的运行错误处理机制。Java中引入了异常和异常类。   Java中的异常处理   一般来讲错误可以在编译时捕获,但是有些错误得在运行期间解决,比如除0等。要考虑到这些方面对可能发生的异常事件作出相应的处理。   Java采用面向对象的方法来处理异常如果一个方法在运行时产生了异常,则这个方法生成代表该异常的一个对象,并把它交给运行时系统,系统会寻找相应的代码来处理异常。   把生成异常对象并把它交给运行时系统的过程称为抛出(throw)异常。系统在方法的调用栈中查找直到找到包含相应异常处理的方法为止,这一过程称为捕获(catch)异常。   Throwable与Exception   The Throwable class is the superclass of all   errors and exceptions in the Java language.   Java中的异常类都是java.lang.Throwable的子类,派生两个子类:Error and Exception。其中Error系统保留,Exception供应用程序使用。   Error:JVM系统内部错误、资源耗尽等,应用程序不处理Error类。   Exception:其他编程错误等一般性问题。一般所说的异常都指Exception及其子类。   Exception也有自己的方法和属性。它有两个构造方法:   public Exception();   public Exception(String s);   s通常是对该例外所对应的错误的描述。   Exception类还继承了父类的方法,常用的:   public String toString(); 返回描述当前Exception 类信息的字符串。   public void printStackTrace(); 在当前的输出上打印当前例外对象的堆栈使用轨迹,即程序使用执行了哪些对象、类,使得产生了例外。   系统定义的异常:   Exception的子类分为RuntimeException和非RuntimeException。   前者是一种设计和实现时的问题,如数组越界等,这种异常可以通过编程避免。   后者是在程序运行过程中由环境原因造成的异常。   用户定义的异常   这类异常是由Exception或其子类所派生出来的类,用于处理与具体应用相关的异常。   异常处理   捕获和处理   异常处理机制:   1、程序执行时出现异常,会自动生成一个异常类对象,该对象被提交给java的运行是系统,此过程称为抛出异常,也可由程序强制执行。   2、系统接收到异常对象,会寻找处理这一异常的代码并把当前异常对象交给它处理,该过程称为捕获异常。   3、如果系统找不到可以捕获异常的方法,则运行时系统将终止,程序也会退出。   异常处理   抛出异常   所有的系统定义的运行异常都可以由系统自动抛出。   用户程序自定义的异常不能由系统自动抛出,必须 throw语句定义何种情况算是产生了异常对应的错误,并且应该抛出这个异常类的对象。   throw 异常对象;   注:1、throw语句一般被定义为满足一定条件时执行。如放在if分支中。   2、使用throw语句的方法,或者调用其他类的有异常抛出的方法时,应在方法头定义中增

    02

    分享:Linux标准输入/输出和重定向

    1. 标准输入与输出 我们知道,执行一个shell命令行时通常会自动打开三个标准文件,即标准输入文件(stdin),通常对应终端的键盘;标准输出文件(stdout)和标准错误输出文件(stderr),这两个文件都对应终端的屏幕。进程将从标准输入文件中得到输入数据,将正常输出数据输出到标准输出文件,而将错误信息送到标准错误文件中。 我们以cat命令为例,cat命令的功能是从命令行给出的文件中读取数据,并将这些数据直接送到标准输出。若使用如下命令: $ cat config 将会把文件config的内容依次显示到屏幕上。但是,如果cat的命令行中没有参数,它就会从标准输入中读取数据,并将其送到标准输出。例如: $ cat Hello world Hello world Bye Bye $ 用户输入的每一行都立刻被cat命令输出到屏幕上。 另一个例子,命令sort按行读入文件正文(当命令行中没有给出文件名时,表示从标准输入读入),将其排序,并将结果送到标准输出。下面的例子是从标准输入读入一个采购单,并将其排序。 $ sort bananas carrots apples apples bananas carrots $ 这时我们在屏幕上得到了已排序的采购单。 直接使用标准输入/输出文件存在以下问题: 输入数据从终端输入时,用户费了半天劲输入的数据只能用一次。下次再想用这些数据时就得重新输入。而且在终端上输入时,若输入有误修改起来不是很方便。 输出到终端屏幕上的信息只能看不能动。我们无法对此输出作更多处理,如将输出作为另一命令的输入进行进一步的处理等。 为了解决上述问题,Linux系统为输入、输出的传送引入了另外两种机制,即输入/输出重定向和管道。 输入重定向 输入重定向是指把命令(或可执行程序)的标准输入重定向到指定的文件中。也就是说,输入可以不来自键盘,而来自一个指定的文件。所以说,输入重定向主要用于改变一个命令的输入源,特别是改变那些需要大量输入的输入源。 例如,命令wc统计指定文件包含的行数、单词数和字符数。如果仅在命令行上键入: $ wc wc将等待用户告诉它统计什么,这时shell就好象死了一样,从键盘键入的所有文本都出现在屏幕上,但并没有什么结果,直至按下<ctrl+d>,

    03

    Linux xargs 命令

    xargs 是给命令传递参数的一个过滤器,也是组合多个命令的一个工具。xargs 可以将管道或标准输入(stdin)数据转换成命令行参数,也能够从文件的输出中读取数据。xargs 也可以将单行或多行文本输入转换为其他格式,例如多行变单行,单行变多行。xargs 默认的命令是 echo,这意味着通过管道传递给 xargs 的输入将会包含换行和空白,不过通过 xargs 的处理,换行和空白将被空格取代。xargs 是一个强有力的命令,它能够捕获一个命令的输出,然后传递给另外一个命令。之所以能用到这个命令,关键是由于很多命令不支持|管道来传递参数,而日常工作中有有这个必要,所以就有了 xargs 命令,例如:

    01
    领券