gpt4 book ai didi

c# - StandardOutput 流上已在进行异步读取操作的异常

转载 作者:太空狗 更新时间:2023-10-29 23:52:20 26 4
gpt4 key购买 nike

我正在尝试使用 C# 来控制命令行应用程序后台,可以在这里下载:http://www.ti.com/litv/zip/spmc015b

这是一个电机电压控制的应用程序,当我进入应用程序时,如“b.exe -c 1”,控制台似乎是一种阻塞模型。此应用程序中的每个命令都以“#”符号开头。在这里查看图片:

http://i46.tinypic.com/zx5edv.jpg

我想做的是,使用 StandardInput.WriteLine("stat vout");来测量电压。这将向控制台后台发送一个“stat vout”命令,并理想地返回一个电压值。在这张照片中,它返回一些帮助提示。一直以来,它仍然处于阻塞模式。

我想用 StandardOutput.ReadLine() 获取返回信息;但失败了。如果 ReadToEnd() 那么我的程序被卡住,因为这个应用程序永远不会返回到阻塞的标准控制台。

当我尝试 BeginOutputReadLine(); OutputDataReceived事件可以真正的获取控制台返回的消息,如stat [vout|vbus|fault]的图片。但它在我的单线程程序中受限。

我目前的情况是,我在 WinForms 中使用 System.Timers.Timers 并且每一秒都会发送一个“stat vout2”命令来读取电压,并希望得到返回值。

但是,System.Timers.Timers是异步的,所以当我在这个Timers线程中调用BeginOutputReadLine()时,提示“一个异步读取操作已经在流上开始”。与此同时,正如我在上面所演示的那样,我不能使用像 ReadLine() 这样的同步方法来获取值。

那我现在该怎么办呢?我确实需要在多线程模式下运行此命令行应用程序。

非常感谢,祝大家周末愉快。

--更新于 4 月 28 日 19:18 CST这是相关的源代码:

其中一个 WinFroms 按钮将 Start() SystemClock 类,然后每秒执行一次 Timing.Measuring()。 TimingController类会根据系统时钟在一秒内同时调用GetVoltage()和GetCurrent(),测量电压和电流。

在测量类中,StandardInput.WriteLine("stat vout2");从控制台应用程序获取电压,以及 StandardInput.WriteLine("stat cur");得到电流。由于 StandardOutput 不起作用,它们都使用 BeginOutputReadLine() 来获取结果。

我使用 isOutputObtained 标志来指示是否返回数据。每次读完,我确实调用了CancelOutputRead();取消异步读取。

但它仍然给我“异步读取操作已经在 StandardOutput 流上进行”的错误异常

public class SystemClock
{
TimingController Timing = new TimingController();
private Timer TimerSystemClock;

public SystemClock()
{
TimerSystemClock = new Timer();
TimerSystemClock.Interval = 1000;
TimerSystemClock.AutoReset = true;
TimerSystemClock.Elapsed += new ElapsedEventHandler(TimerSystemClock_Elapsed);
TimerSystemClock.Enabled = false;
Timing.ClockInstance = this;
}

public void Start()
{
TimerSystemClock.Enabled = true;
}

void TimerSystemClock_Elapsed(object sender, ElapsedEventArgs e)
{
Timing.Measuring();
}
}

public class TimingController
{
// Get singleton of Measurement Class
Measurement Measure = Measurement.GetInstance();

public SystemClock ClockInstance
{
get { return Clock; }
set { Clock = value; }
}

private void Measuring()
{
CurrentVoltage = Measure.GetVoltage();
CurrentCurrent = Measure.GetCurrent();
}
}

public sealed class Measurement
{
// Singleton
public static readonly Measurement instance = new Measurement();
public static Measurement GetInstance()
{
return instance;
}

private Process ProcMeasuring = new Process();
double measureValue
bool isOutputObtained;

private Measurement()
{
ProcMeasuring.StartInfo.FileName = "b.exe";
ProcMeasuring.StartInfo.Arguments = "-c 1";
ProcMeasuring.StartInfo.WorkingDirectory = Directory.GetCurrentDirectory();
ProcMeasuring.StartInfo.UseShellExecute = false;
ProcMeasuring.StartInfo.RedirectStandardInput = true;
ProcMeasuring.StartInfo.RedirectStandardOutput = true;
ProcMeasuring.StartInfo.RedirectStandardError = true;
ProcMeasuring.StartInfo.CreateNoWindow = true;
ProcMeasuring.OutputDataReceived += new DataReceivedEventHandler(MeasurementOutputHandler);
}

public double GetVoltage(Machine machine)
{
isOutputObtained = false;

ProcMeasuring.StandardInput.WriteLine("stat vout2");
ProcMeasuring.BeginOutputReadLine();

while (!isOutputObtained)
{
isOutputObtained = true;
}

return measureValue;
}

public double GetCurrent(Machine machine)
{
isOutputObtained = false;

ProcMeasuring.StandardInput.WriteLine("stat cur");
ProcMeasuring.BeginOutputReadLine();

while (!isOutputObtained)
{
isOutputObtained = true;
}

return measureValue;
}

private void MeasurementOutputHandler(object sendingProcess, DataReceivedEventArgs outLine)
{
if (!String.IsNullOrEmpty(outLine.Data) && (outLine.Data != "# "))
{
measureCurrentValue = Convert.ToDouble(outLine.Data);
isOutputObtained = true;

ProcMeasuring.CancelOutputRead();
}
}
}

最佳答案

我认为您有两个选择。

如果因为前一个花费的时间太长而错过一个样本是可以的,那么设置一个标志以指示您已经在采样。

public class TimingController  
{
// Get singleton of Measurement Class
Measurement Measure = Measurement.GetInstance();

bool isMeasuring = false;

public SystemClock ClockInstance
{
get { return Clock; }
set { Clock = value; }
}

private void Measuring()
{
if(isMeasuring) return;

isMeasuring = true;
CurrentVoltage = Measure.GetVoltage();
CurrentCurrent = Measure.GetCurrent();
isMeasuring = false;
}
}

如果不能错过一个间隔,您可以尝试为每次调用创建一个新的流程对象,而不是重复使用现有的流程。如果命令需要很长时间才能完成,这可能会创建很多 child 。

关于c# - StandardOutput 流上已在进行异步读取操作的异常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10362207/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com