5

我正在尝试使用 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() 这样的同步方法来获取值。

那我现在该怎么办?我真的需要在多线程模式下运行这个命令行应用程序。

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

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

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

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

我使用 isOutputObtained 标志来指示数据是否返回。每次阅读完成时,我都会调用 CancelOutputRead(); 取消异步读取。

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

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();
        }
    }
}
4

1 回答 1

0

我认为你有两个选择。

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

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;
    }  
} 

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

于 2012-05-04T14:11:08.413 回答