使用 C# 捕获进程输出
使用 C# 捕获进程输出
Intro
很多时候我们可能会需要执行一段命令获取一个输出,遇到的比较典型的就是之前我们需要用 FFMpeg
实现视频的编码压缩水印等一系列操作,当时使用的是 FFMpegCore
这个类库,这个类库的实现原理是启动另外一个进程,启动 ffmpeg 并传递相应的处理参数,并根据进程输出获取处理进度
为了方便使用,实现了两个帮助类来方便的获取进程的输出,分别是 ProcessExecutor
和 CommandRunner
,前者更为灵活,可以通过事件添加自己的额外事件订阅处理,后者为简化版,主要是只获取输出的场景,两者的实现原理大体是一样的,启动一个 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
- https://github.com/rosenbjerg/FFMpegCore
- https://github.com/WeihanLi/WeihanLi.Common/blob/dev/src/WeihanLi.Common/Helpers/ProcessExecutor.cs
- https://github.com/WeihanLi/WeihanLi.Common/blob/dev/test/WeihanLi.Common.Test/HelpersTest/ProcessExecutorTest.cs
- https://github.com/WeihanLi/WeihanLi.Common/blob/dev/test/WeihanLi.Common.Test/HelpersTest/CommandRunnerTest.cs
使用 C# 捕获进程输出的更多相关文章
- [linux] C语言Linux系统编程-捕获进程信号
typedef void( *sighandler_t)(int); 1.用typedef给类型起一个别名. 2.为函数指针类型定义别名, 3.函数指针(指向函数的指针) sighandler_t s ...
- sql存储过程异常捕获并输出例子还有不输出过程里面判断异常 例子
编程的异常处理很重要,当然Sql语句中存储过程的异常处理也很重要,明确的异常提示能够快速的找到问题的根源,节省很多时间. 下面,我就以一个插入数据为例来说明Sql Server中的存储过程怎么捕获异常 ...
- shell无法捕获程序输出的问题
dir_name=`echo ~gtp` 获取的用户目录为/ dir_name=`echo ~gtp 2>&1` 这样就可以获取到了 参考网址:https://blog.csdn.net ...
- 一条命令,根据进程名判断有进程输出up,无进程无输出
这个研究了好一会, 由于开发需要,提供的命令. shell命令,可以按照分号分割,也可以按照换行符分割.如果想一行写入多个命令,可以通过“';”分割. a=`ps -ef | grep nginx | ...
- DirectX屏幕捕获和输出视频
#include <Windows.h> #include <mfapi.h> #include <mfidl.h> #include <Mfreadwrit ...
- python中如何用sys.excepthook来对全局异常进行捕获、显示及输出到error日志中
使用sys.excepthook函数进行全局异常的获取. 1. 使用MessageDialog实现异常显示: 2. 使用logger把捕获的异常信息输出到日志中: 步骤:定义异常处理函数, 并使用该函 ...
- Java示例:如何执行进程并读取输出
下面是一个例子,演示如何执行一个进程(类似于在命令行下键入命令),读取进程执行的输出,并根据进程的返回值判断是否执行成功.一般来说,进程返回 0 表示执行成功,其他值表示失败. import java ...
- stm32f103_高级定时器——输入捕获/输出比较中断+pwm=spwm生成
****************************首选我们了解一下它们的功能吧********************************************************** ...
- 内核创建的用户进程printf不能输出一问的研究
转:http://www.360doc.com/content/09/0315/10/26398_2812414.shtml 一:前言上个星期同事无意间说起,在用核中创建的用户空间进程中,使用prin ...
随机推荐
- 5.19 省选模拟赛 小B的夏令营 概率 dp 前缀和优化dp
LINK:小B的夏令营 这道题是以前从没见过的优化dp的方法 不过也在情理之中. 注意读题 千万不要像我这个sb一样 考完连题意都不知道是啥. 一个长方形 要求从上到下联通的概率. 容易发现 K天只是 ...
- Pintech品致全新多功能MDO 704E系列示波器全新推出
2020年 7月,Pintech品致全新推出推出首款具有多个模拟通道和多个数字通道的示波器.每个模拟通道带宽为200 MHz,每个模拟通道采样率同时达1 GSa/s,在一台仪器中,实现精确.可重复的. ...
- 关于 ORA-01033: ORACLE initialization or shutdown in progress
第一步: 这个错误首先查看服务进程是否正常启动: 第二步: 一般情况下第一步都没问题,问题出在可能误删了日志文件: 当然可能不是你删除的,可能被某些清理软件删除的: 或者是其他情况导致日志出错 ...
- C++文件操作和模板
1.数据层次 位 bit 字节 byte 域/记录 将所有记录顺序地写入一个文件---->顺序文件:一个有限字符构成的顺序字符流 C++标准库中:ifsteam,ofstream,fstream ...
- 串行&并行&并发,同步&异步
1. 串行&并行&并发 1.1 串行 这个非常好理解,字面意思,像串成一个串一样,顺序执行 上一个没执行完的话,后面的就必须无条件等待 一般情况就是一个线程里:任务一个接一个执行,类似 ...
- windows下Nginx+RTMP部署
在windows下部署参考博客:https://blog.csdn.net/l1028386804/article/details/80022437 参考这个部署完成后,一般没有什么问题,可以看到视频 ...
- Meow 攻击会删除不安全(开放的)的Elasticsearch(及MongoDB) 索引,然后建一堆以Meow结尾的奇奇怪怪的索引(如:m3egspncll-meow)
07月29日,早上照例一来,先连接Elasticsearch查看日志[禁止转载,by @CoderBaby],结果,咦,什么情况,相关索引被删除了,产生了一堆以Meow开头的奇奇怪怪的索引,如下图: ...
- JavaScript的函数和作用域闭包
1. 函数 1.1 定义函数 function add(x, y){ return x + y; } 上述函数定义如下: 关键字function指出这是一个函数定义: add是函数的名称: (x, y ...
- CRF
- Go之Gorm和BeegoORM简介及配置使用
简介 ORM Object-Relationl Mapping, 它的作用是映射数据库和对象之间的关系,方便我们在实现数据库操作的时候不用去写复杂的sql语句,把对数据库的操作上升到对于对象的操作 G ...