6

我正在使用以下代码从 com 端口读取值:

Private port As New SerialPort("COM13", 9600, Parity.None, 8, StopBits.One)

Private Sub port_DataReceived(ByVal sender As Object, ByVal e As SerialDataReceivedEventArgs)
    Debug.Print(port.ReadExisting())
End Sub

Private Sub Form1_Load(ByVal sender As System.Object, ByVal e As System.EventArgs) Handles MyBase.Load
    AddHandler port.DataReceived, New SerialDataReceivedEventHandler(AddressOf port_DataReceived)
    port.Open()
End Sub

这工作得很好,但时不时地它不会获取所有数据,并且返回结果是两个字符串而不是一个。

一个例子是,如果 com 端口通过单词“HELLO2YOU”发送它看起来像:

HEL
LO2YOU

或者

HELLO2
YOU

我如何在其中放置一个缓冲区,以确保在显示之前读取所有数据?

谢谢!

4

2 回答 2

9

您必须将串行端口通信视为流数据。每当您收到数据时,您都必须期望它可能是完整的消息、仅是部分消息或多条消息。这完全取决于数据进入的速度以及应用程序能够从队列中读取的速度。因此,您认为需要缓冲区是正确的。但是,您可能还没有意识到,严格来说,无法通过串行端口知道每条消息的开始和结束位置。这必须通过发送方和接收方之间商定的协议来处理。例如,许多人使用标准的文本开始 (STX) 和文本结束 (ETX) 字符来指示每条消息发送的开始和结束。这样,当您收到数据时,您就可以知道您何时收到了完整的消息。

例如,如果你使用 STX 和 ETX 字符,你可以创建一个这样的类:

Public Class DataBuffer
    Private ReadOnly _startOfText As String = ASCII.GetChars(New Byte() {2})
    Private ReadOnly _endOfText As String = ASCII.GetChars(New Byte() {4})

    Public Event MessageReceived(ByVal message As String)
    Public Event DataIgnored(ByVal text As String)

    Private _buffer As StringBuilder = New StringBuilder

    Public Sub AppendText(ByVal text As String)
        _buffer.Append(text)
        While processBuffer(_buffer)
        End While
    End Sub

    Private Function processBuffer(ByVal buffer As StringBuilder) As Boolean
        Dim foundSomethingToProcess As Boolean = False
        Dim current As String = buffer.ToString()
        Dim stxPosition As Integer = current.IndexOf(_startOfText)
        Dim etxPosition As Integer = current.IndexOf(_endOfText)
        If (stxPosition >= 0) And (etxPosition >= 0) And (etxPosition > stxPosition) Then
            Dim messageText As String = current.Substring(0, etxPosition + 1)
            buffer.Remove(0, messageText.Length)
            If stxPosition > 0 Then
                RaiseEvent DataIgnored(messageText.Substring(0, stxPosition))
                messageText = messageText.Substring(stxPosition)
            End If
            RaiseEvent MessageReceived(messageText)
            foundSomethingToProcess = True
        ElseIf (stxPosition = -1) And (current.Length <> 0) Then
            buffer.Remove(0, current.Length)
            RaiseEvent DataIgnored(current)
            foundSomethingToProcess = True
        End If
        Return foundSomethingToProcess
    End Function


    Public Sub Flush()
        If _buffer.Length <> 0 Then
            RaiseEvent DataIgnored(_buffer.ToString())
        End If
    End Sub
End Class

我还应该提到,在通信协议中,通常有一个校验和字节,您可以通过该字节确定消息在发送方和接收方之间的传输过程中是否损坏。

于 2012-07-12T19:13:26.990 回答
3

这是很正常的,串口是非常慢的设备。使用像 9600 这样的波特率并且机器不会陷入太多困境,当您使用 ReadExisting() 时,您只会从端口获得一两个字节。Debug.Print() 输出一个行终止符,因此您会看到收到的任何内容都被分解成碎片。

修复它的最简单方法是改用 ReadLine()。这要求设备在行尾发送一个特殊字符,该字符与 SerialPort.NewLine 属性值匹配。这很常见,换行符是样板。

如果没有,那么您将需要某种其他类型的缓冲方案。

于 2012-07-12T19:14:50.323 回答