使用 C# 捕获进程输出

Intro

很多时候我们可能会需要执行一段命令获取一个输出,遇到的比较典型的就是之前我们需要用 FFMpeg 实现视频的编码压缩水印等一系列操作,当时使用的是 FFMpegCore 这个类库,这个类库的实现原理是启动另外一个进程,启动 ffmpeg 并传递相应的处理参数,并根据进程输出获取处理进度

为了方便使用,实现了两个帮助类来方便的获取进程的输出,分别是 ProcessExecutorCommandRunner,前者更为灵活,可以通过事件添加自己的额外事件订阅处理,后者为简化版,主要是只获取输出的场景,两者的实现原理大体是一样的,启动一个 Process,并监听其输出事件获取输出

ProcessExecutor

使用示例,这个示例是获取保存 nuget 包的路径的一个示例:

using var executor = new ProcessExecutor("dotnet", "nuget locals global-packages -l");
var folder = string.Empty;
executor.OnOutputDataReceived += (sender, str) =>
{
if(str is null)
return; Console.WriteLine(str); if(str.StartsWith("global-packages:"))
{
folder = str.Substring("global-packages:".Length).Trim();
}
};
executor.Execute(); Console.WriteLine(folder);

ProcessExecutor 实现代码如下:

public class ProcessExecutor : IDisposable
{
public event EventHandler<int> OnExited; public event EventHandler<string> OnOutputDataReceived; public event EventHandler<string> OnErrorDataReceived; protected readonly Process _process; protected bool _started; public ProcessExecutor(string exePath) : this(new ProcessStartInfo(exePath))
{
} public ProcessExecutor(string exePath, string arguments) : this(new ProcessStartInfo(exePath, arguments))
{
} public ProcessExecutor(ProcessStartInfo startInfo)
{
_process = new Process()
{
StartInfo = startInfo,
EnableRaisingEvents = true,
};
_process.StartInfo.UseShellExecute = false;
_process.StartInfo.CreateNoWindow = true;
_process.StartInfo.RedirectStandardOutput = true;
_process.StartInfo.RedirectStandardInput = true;
_process.StartInfo.RedirectStandardError = true;
} protected virtual void InitializeEvents()
{
_process.OutputDataReceived += (sender, args) =>
{
if (args.Data != null)
{
OnOutputDataReceived?.Invoke(sender, args.Data);
}
};
_process.ErrorDataReceived += (sender, args) =>
{
if (args.Data != null)
{
OnErrorDataReceived?.Invoke(sender, args.Data);
}
};
_process.Exited += (sender, args) =>
{
if (sender is Process process)
{
OnExited?.Invoke(sender, process.ExitCode);
}
else
{
OnExited?.Invoke(sender, _process.ExitCode);
}
};
} protected virtual void Start()
{
if (_started)
{
return;
}
_started = true; _process.Start();
_process.BeginOutputReadLine();
_process.BeginErrorReadLine();
_process.WaitForExit();
} public async virtual Task SendInput(string input)
{
try
{
await _process.StandardInput.WriteAsync(input!);
}
catch (Exception e)
{
OnErrorDataReceived?.Invoke(_process, e.ToString());
}
} public virtual int Execute()
{
InitializeEvents();
Start();
return _process.ExitCode;
} public virtual async Task<int> ExecuteAsync()
{
InitializeEvents();
return await Task.Run(() =>
{
Start();
return _process.ExitCode;
}).ConfigureAwait(false);
} public virtual void Dispose()
{
_process.Dispose();
OnExited = null;
OnOutputDataReceived = null;
OnErrorDataReceived = null;
}
}

CommandExecutor

上面的这种方式比较灵活但有些繁琐,于是有了下面这个版本

使用示例:

[Fact]
public void HostNameTest()
{
if (!RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
{
return;
} var result = CommandRunner.ExecuteAndCapture("hostname"); var hostName = Dns.GetHostName();
Assert.Equal(hostName, result.StandardOut.TrimEnd());
Assert.Equal(0, result.ExitCode);
}

实现源码:

public static class CommandRunner
{
public static int Execute(string commandPath, string arguments = null, string workingDirectory = null)
{
using var process = new Process()
{
StartInfo = new ProcessStartInfo(commandPath, arguments ?? string.Empty)
{
UseShellExecute = false,
CreateNoWindow = true, WorkingDirectory = workingDirectory ?? Environment.CurrentDirectory
}
}; process.Start();
process.WaitForExit();
return process.ExitCode;
} public static CommandResult ExecuteAndCapture(string commandPath, string arguments = null, string workingDirectory = null)
{
using var process = new Process()
{
StartInfo = new ProcessStartInfo(commandPath, arguments ?? string.Empty)
{
UseShellExecute = false,
CreateNoWindow = true, RedirectStandardOutput = true,
RedirectStandardError = true, WorkingDirectory = workingDirectory ?? Environment.CurrentDirectory
}
};
process.Start();
var standardOut = process.StandardOutput.ReadToEnd();
var standardError = process.StandardError.ReadToEnd();
process.WaitForExit();
return new CommandResult(process.ExitCode, standardOut, standardError);
}
} public sealed class CommandResult
{
public CommandResult(int exitCode, string standardOut, string standardError)
{
ExitCode = exitCode;
StandardOut = standardOut;
StandardError = standardError;
} public string StandardOut { get; }
public string StandardError { get; }
public int ExitCode { get; }
}

More

如果只要执行命令获取是否执行成功则使用 CommandRunner.Execute 即可,只获取输出和是否成功可以用 CommandRunner.ExecuteAndCapture 方法,如果想要进一步的添加事件订阅则使用 ProcessExecutor

Reference

使用 C# 捕获进程输出的更多相关文章

  1. [linux] C语言Linux系统编程-捕获进程信号

    typedef void( *sighandler_t)(int); 1.用typedef给类型起一个别名. 2.为函数指针类型定义别名, 3.函数指针(指向函数的指针) sighandler_t s ...

  2. sql存储过程异常捕获并输出例子还有不输出过程里面判断异常 例子

    编程的异常处理很重要,当然Sql语句中存储过程的异常处理也很重要,明确的异常提示能够快速的找到问题的根源,节省很多时间. 下面,我就以一个插入数据为例来说明Sql Server中的存储过程怎么捕获异常 ...

  3. shell无法捕获程序输出的问题

    dir_name=`echo ~gtp` 获取的用户目录为/ dir_name=`echo ~gtp 2>&1` 这样就可以获取到了 参考网址:https://blog.csdn.net ...

  4. 一条命令,根据进程名判断有进程输出up,无进程无输出

    这个研究了好一会, 由于开发需要,提供的命令. shell命令,可以按照分号分割,也可以按照换行符分割.如果想一行写入多个命令,可以通过“';”分割. a=`ps -ef | grep nginx | ...

  5. DirectX屏幕捕获和输出视频

    #include <Windows.h> #include <mfapi.h> #include <mfidl.h> #include <Mfreadwrit ...

  6. python中如何用sys.excepthook来对全局异常进行捕获、显示及输出到error日志中

    使用sys.excepthook函数进行全局异常的获取. 1. 使用MessageDialog实现异常显示: 2. 使用logger把捕获的异常信息输出到日志中: 步骤:定义异常处理函数, 并使用该函 ...

  7. Java示例:如何执行进程并读取输出

    下面是一个例子,演示如何执行一个进程(类似于在命令行下键入命令),读取进程执行的输出,并根据进程的返回值判断是否执行成功.一般来说,进程返回 0 表示执行成功,其他值表示失败. import java ...

  8. stm32f103_高级定时器——输入捕获/输出比较中断+pwm=spwm生成

    ****************************首选我们了解一下它们的功能吧********************************************************** ...

  9. 内核创建的用户进程printf不能输出一问的研究

    转:http://www.360doc.com/content/09/0315/10/26398_2812414.shtml 一:前言上个星期同事无意间说起,在用核中创建的用户空间进程中,使用prin ...

随机推荐

  1. PHP imageaffinematrixconcat - 连接两个矩阵

    imageaffinematrixconcat — 连接两个矩阵.高佣联盟 www.cgewang.com 语法 array imageaffinematrixconcat ( array $m1 , ...

  2. PHP strcasecmp() 函数

    实例 比较两个字符串(不区分大小写): <?php高佣联盟 www.cgewang.comecho strcasecmp("Hello world!","HELLO ...

  3. C 语言学习 --2

    memset Declaration: void *memset(void *str, int c, size_t n); Copies the character c (an unsigned ch ...

  4. RNN神经网络产生梯度消失和梯度爆炸的原因及解决方案

    1.RNN模型结构 循环神经网络RNN(Recurrent Neural Network)会记忆之前的信息,并利用之前的信息影响后面结点的输出.也就是说,循环神经网络的隐藏层之间的结点是有连接的,隐藏 ...

  5. VulnHub靶场学习_HA: Natraj

    HA: Natraj Vulnhub靶场 下载地址:https://www.vulnhub.com/entry/ha-natraj,489/ 背景: Nataraj is a dancing avat ...

  6. hashCode竟然不是根据对象内存地址生成的?还对内存泄漏与偏向锁有影响?

    起因 起因是群里的一位童鞋突然问了这么问题: 如果重写 equals 不重写 hashcode 会有什么影响? 这个问题从上午10:45 开始陆续讨论,到下午15:39 接近尾声 (忽略这形同虚设的马 ...

  7. Springboot+swagger2.7集成开发

    Springboot+swagger2.7集成开发 本篇文章是介绍最新的springboot和swagger2.7集成开发和2.0稍微有一些出入: Springboot集成环境配置 Swagger2. ...

  8. C#LeetCode刷题之#581-最短无序连续子数组( Shortest Unsorted Continuous Subarray)

    问题 给定一个整数数组,你需要寻找一个连续的子数组,如果对这个子数组进行升序排序,那么整个数组都会变为升序排序. 你找到的子数组应是最短的,请输出它的长度. 输入: [2, 6, 4, 8, 10, ...

  9. Vuex mapAction的基本使用

    mapAction-store中的异步方法 import Vue from 'vue' import Vuex from 'vuex' Vue.use(Vuex) export default new ...

  10. Scss 定义内层class的简单写法

    如果定义样式的时候,内层样式名称和外层保持一致的话可以简写如下 如果一个样式下有相关的其他样式可以使用 &-xxx 来简写, <template> <div class=&q ...