C# - 通过串行端口从设备获取所有数据并检测控制字符(ACK,SOH,...)

问题描述 投票:2回答:1

我可以轻松地从设备接收数据的响应,并使用Serial.ReadExisting()在Textbox中显示。显示数据时,问题是某些控制字符(ACK,SOH,ETX,...)无法打印。

我尝试使用下面的代码从Serial响应中检测控制字符,但在比较中出现了错误。

代码是:

public bool read_port(ref string rs232data, int timeout)
    {
        try
        {
            int index_read = 0;
            int total_read = 0;
            int i = 0;
            int delay_ms = 1;
            byte[] buffer = new byte[serialPort1.ReadBufferSize];
            rs232data = "";
            string buffer_str = "";

            if ((serialPort1.IsOpen == false))
            {
                #if DEBUG
                MessageBox.Show("COM port is not opening");
                #endif
                return false;
            }
            do
            {
                Application.DoEvents();
                PauseForMilliSeconds(delay_ms); //System.Threading.Thread.Sleep(delay_ms);
                i++;

                //serial port receive data
                if (serialPort1.BytesToRead <= 0) continue;
                if (i > 1) i--;
                index_read = serialPort1.Read(buffer, total_read, serialPort1.BytesToRead);
                total_read += index_read;
                buffer_str = ToStringLsb(buffer);

                //// Find a Start byte
                int ack_index = buffer_str.LastIndexOf(ToStringLsb(ACK));
                int nak_index = buffer_str.LastIndexOf(ToStringLsb(NAK));

                //Find a Stop byte
                int stop_index_etx = buffer_str.LastIndexOf(ToStringLsb(ETX));
                int stop_index_lf = buffer_str.LastIndexOf(ToStringLsb(LF));

                if ((stop_index_etx < 0) && (stop_index_lf < 0) && (ack_index < 0)) continue; // Can't find a byte Stop


                if (stop_index_etx > 0)
                {
                    if ((stop_index_etx + 1) < total_read)
                    {
                        stop_index_etx++;
                    }
                    else
                    {
                        if (buffer[total_read - 2] == ETX)
                        {
                            stop_index_etx--;
                        }
                        else continue;
                    }
                }

                rs232data = buffer_str.Substring(0, total_read);
                return true;
            }
            while (i < timeout);
            return false;
        }
        catch (Exception ex)
        {
            MessageBox.Show(ex.Message);
            return false;
        }
    }

    public static string ToStringLsb(byte bytevalue)
    {
        try
        {
            byte[] bytearray = { bytevalue };
            return System.Text.Encoding.GetEncoding("iso-8859-1").GetString(bytearray);
        }
        catch (System.Exception ex)
        {
            MessageBox.Show(ex.Message);
            return "";
        }
    }

    public static string ToStringLsb(byte[] bytearray)
    {
        return System.Text.Encoding.GetEncoding("iso-8859-1").GetString(bytearray);
    }
c# serial-port
1个回答
0
投票

您需要查看ASCII映射并确定哪些值是您要查找的控件值。

ASCII Map

例如,我将在我的代码中定义以下控制字符,然后在DataReceived事件中,您可以查找这些字符。

char ACK = (char)6;
char NAK = (char)21;
char SOH = (char)1;
char LF = (char)10;
StringBuilder sb = new StringBuilder();

private void serialPort1_DataReceived(object sender, System.IO.Ports.SerialDataReceivedEventArgs e)
{
    string Data = serialPort1.ReadExisting();

    foreach (char c in Data)
    {
        if (c == LF)
        {
            sb.Append(c);

            CurrentLine = sb.ToString();
            sb.Clear();

            //parse CurrentLine here or print it to textbox
        }
        if (c == ACK)
        {
            sb.Append("<ACK>"); //or whatever you want to print
        }
        else
        {
            sb.Append(c);
        }
    }
}

编辑:在下面的评论中回答您的一些问题。 DataReceived事件在它的缓冲区中获取字符时会触发,但它可以触发,同时只获得一半的消息。因此,在使用此事件时,您必须构建一个字符串,直到您知道您拥有整个消息。在上面的例子中,我假设LF(换行)表示我有整个消息。对于你,你会使用你正在寻找的任何字符来标记你的信息的结尾(在你的情况下可能是ACK?)。您可以选择是否将该字符附加到字符串中,这取决于您的要求。在我的例子中,你可以看到我附加LF,但你可以很容易地把那条线拿出来。

© www.soinside.com 2019 - 2024. All rights reserved.