繁体   English   中英

如何使用串口数据编写自己的Get / Request函数?

[英]How to write your own Get / Request function with serial port data?

我有一个与串口通讯的类,我想模仿一个Get函数。 问题是,串行端口输入的原始数据可能会或可能不会。 还必须对其进行处理和解析。 例如,如果我要获取一个Person对象,则串行端口类会这样做。

  1. 一种发送请求给Person的方法。

  2. 串行设备上发生了某些事情...它可能会响应,但可能不会。

  3. 字节通过事件处理程序进入,该事件处理程序处理并解析所有数据,最后是Person对象。

我想将所有内容包装到一个“ GetPerson”函数中,但是在连接第1步和第3步时遇到困难。第3步是一个始终运行的函数,更像是一个通用解析器。

有人可以帮我设计/结构吗?

编辑:同样,我也尝试过使用类变量并通过while语句进行检查,如下所示,但是出于任何原因,它都不会在while语句中检查变量:

private Person person;

public Person GetPerson()
{
   ...
   while (person == null) {}
   return person;
}

据我了解,您需要将EAP转换为TAP 换句话说,以异步\\等待方式等待某些数据接收事件。 您可以TaskComplettionSource<>使用TaskComplettionSource<> 基于本文,我试图想象设计合适的解决方案:

public class PersonSerialPort : IDisposable
{
    private readonly SerialPort port;

    /// <summary>
    /// Timeout in milliseconds
    /// </summary>
    private const int Timeout = 5000;

    public PersonSerialPort()
    {
        // port initializing here
        port = new SerialPort(/*your parameters here*/);
        port.Open();
    }

    public async Task<Person> GetPerson()
    {
        // set up the task completion source
        var tcs = new TaskCompletionSource<Person>();

        // handler of DataReceived event of port
        var handler = default(SerialDataReceivedEventHandler);
        handler = (sender, eventArgs) =>
        {
            try
            {
                Person result = new Person();

                // some logic for filling Person fields
                // or set it null or whatever you need
                // you are free to not set result and wait for next event fired too

                tcs.SetResult(result);

            }
            finally
            {
                port.DataReceived -= handler;
            }
        };
        port.DataReceived += handler;

        // send request for person
        port.Write("Give a person number 1");

        if (await Task.WhenAny(tcs.Task, Task.Delay(Timeout)) == tcs.Task)
        {
            return tcs.Task.Result;
        }
        else
        {
            port.DataReceived -= handler;
            throw new TimeoutException("Timeout has expired");
        }
    }

    public void Dispose()
    {
        port?.Dispose();
    }
}

UPD:已添加超时逻辑,另请参阅此问题

希望能帮助到你。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM