C#SerialPort - 混合具有不同波特率的端口的问题

Gra*_*ral 5 c# serial-port baud-rate

我有两个设备,我想通过串行接口连接,但它们有不兼容的连接.为了解决这个问题,我将它们连接到我的PC上,我正在开发一个C#程序,它将把COM端口X上的流量路由到COM端口Y,反之亦然.

该程序连接到两个COM端口.在数据接收事件处理程序中,我读入传入的数据并将其写入另一个COM端口.为此,我有以下代码:

    private void HandleDataReceived(SerialPort inPort, SerialPort outPort)
    {
        byte[] data = new byte[1];

        while (inPort.BytesToRead > 0)
        {
            // Read the data
            data[0] = (byte)inPort.ReadByte();

            // Write the data
            if (outPort.IsOpen)
            {
                outPort.Write(data, 0, 1);
            }
        }
    }
Run Code Online (Sandbox Code Playgroud)

只要传出的COM端口以高于传入COM端口的波特率运行,该代码就能正常工作.如果传入的COM端口比传出的COM端口快,我开始丢失数据.我不得不纠正这样的代码:

    private void HandleDataReceived(SerialPort inPort, SerialPort outPort)
    {
        byte[] data = new byte[1];

        while (inPort.BytesToRead > 0)
        {
            // Read the data
            data[0] = (byte)inPort.ReadByte();

            // Write the data
            if (outPort.IsOpen)
            {
                outPort.Write(data, 0, 1);
                while (outPort.BytesToWrite > 0);  //<-- Change to fix problem
            }
        }
    }
Run Code Online (Sandbox Code Playgroud)

我不明白为什么我需要那个修复.我是C#的新手(这是我的第一个程序),所以我想知道是否有我遗漏的东西.SerialPort默认为2048字节写缓冲区,我的命令少于10个字节.写缓冲区应该能够缓冲数据,直到它可以写入较慢的COM端口.

总之,我在COM X上接收数据并将数据写入COM Y. COM X以比COM Y更快的波特率连接.为什么写缓冲区中的缓冲不能处理这种差异?为什么我需要等待写缓冲区耗尽以避免数据丢失?

谢谢!

*更新*

如上所述,此代码很容易遇到具有大量和/或快速传入数据传输的溢出情况.我应该写更多关于我的数据流的文章.我期待10赫兹的<10字节命令(<10字节响应).另外,我看到第一个命令失败了.

因此,虽然我知道这个代码不能扩展并且不是最佳的,但我想知道为什么2-4K读/写缓冲区甚至无法处理第一个命令.我想知道是否存在使用我不理解的事件处理程序写入单个字节数据的错误.谢谢.

*更新*

这是失败的一个例子:

假设我的命令是四个字节:0x01 0x02 0x3 0x4.COM X上的设备发送命令.我可以看到C#程序接收四个字节并将它们发送到COM Y上的设备.COM Y上的设备接收两个字节:0x01 0x03.我知道COM Y上的设备是可靠的,所以我想知道这两个字节是如何被丢弃的.

顺便说一句,有人可以告诉我,如果只是回复带有评论的答案或者我是否应该继续编辑原始问题?哪个更有帮助?

mtr*_*trw 0

您应该确保它outPort.WriteBufferSize大于您期望发送的最大缓冲区。此外,在循环中调用ReadByteandWriteByte通常会很慢。如果您将处理程序更改为类似以下内容:

int NumBytes = 20; //or whatever makes sense
byte[] data = new byte[NumBytes];

while (inPort.BytesToRead > 0)
{
    // Read as much data as possible at once
    count = inPort.Read(data, 0, min(NumBytes, inPort.BytesToRead));

    // Write the data
    if (outPort.IsOpen)
    {
        outPort.Write(data, 0, count);
    }
}
Run Code Online (Sandbox Code Playgroud)

这将减少开销,这应该会有所帮助。然后,写入缓冲区将(希望)按照您的预期处理时序。