我使用.Net Framework 2.0 Serialport类在C#2.0中编写了一个简单的应用程序,通过COM1与控制器卡进行通信。

最近出现的问题是Read方法返回的字节不正确。它返回了正确的字节数,只有值不正确。用Delphi编写的类似应用程序仍返回正确的值。

我使用 Portmon 来记录串口上的活动两个应用程序,比较两个日志和那里一些(显然)次要的不同设置,我试图尽可能地模仿Delphi应用程序,但无济于事。

那么,什么会影响Read方法返回的字节值?

两个应用之间的大多数设置都是相同的。

以下是Portmon日志中不同行的列表:

Delphi App:

  

IOCTL_SERIAL_SET_CHAR Serial0 SUCCESS EOF:dc   错误:0 BRK:0 EVT:0 XON:11 XOFF:13
  IOCTL_SERIAL_SET_HANDFLOW Serial0 SUCCESS Shake:0   替换:0 XonLimit:256    XoffLimit:256 IOCTL_SERIAL_SET_TIMEOUTS Serial0 SUCCESS RI:-1    RM:100 RC:1000 WM:100 WC:1000 IOCTL_SERIAL_SET_WAIT_MASK Serial0 SUCCESS面具:   RXCHAR RXFLAG TXEMPTY CTS DSR RLSD   BRK ERR RING RX80FULL

C#App:

  

IOCTL_SERIAL_SET_CHAR Serial0 SUCCESS EOF:1a   错误:0 BRK:0 EVT:1a XON:11 XOFF:13   IOCTL_SERIAL_SET_HANDFLOW Serial0 SUCCESS Shake:0   替换:0 XonLimit:1024    XoffLimit:1024 IOCTL_SERIAL_SET_TIMEOUTS Serial0 SUCCESS RI:-1    RM:-1 RC:1000 WM:0 WC:1000 IOCTL_SERIAL_SET_WAIT_MASK Serial0 SUCCESS面具:   RXCHAR RXFLAG CTS DSR RLSD BRK ERR   RING

更新:

正确返回的字节是:91,1,1,3,48,48,50,69,66,51,70,55,52,93(14字节)。 最后一个值是一个简单的校验和。

返回的错误值为:91,241,254,252,242,146,42,201,51,70,55,52,93(13字节)。

如您所见,返回的第一个和最后五个字节对应。

ErrorReceived事件表示发生了帧错误,这可以解释错误的值。但问题是,当Delphi应用显然没有时,为什么SerialPort会遇到帧错误?

有帮助吗?

解决方案

好吧,好像问题已经解决了(至少目前是这样)。

显然,帧错误导致返回错误值。我编写了一个VB6应用程序,使用MSComm控件,工作正常,并比较了Portmon生成的日志文件。

我发现了以下差异

VB6 App:

  

IOCTL_SERIAL_SET_HANDFLOW Serial0   成功摇:1 替换:0   XonLimit:256 XoffLimit:256

C#App:

  

IOCTL_SERIAL_SET_HANDFLOW Serial0   成功震动:0 替换:0   XonLimit:1024 XoffLimit:1024

如果我设置,我会发现设置 _serialPort.DtrEnable = true C#App生成以下日志条目:

  

IOCTL_SERIAL_SET_HANDFLOW Serial0   成功摇:1 替换:0   XonLimit:1024 XoffLimit:1024

这似乎可以防止框架错误,应用程序似乎工作正常。

其他提示

您是否检查了数据位数,停止位和奇偶校验的设置?

奇偶校验位是一种错误检测机制。例如:如果使用7个数据位和一个奇偶校验位发送,则第8位将用于检测位反转错误。如果接收器需要8个数据位且没有奇偶校验位,则结果将是乱码。

不幸的是,您没有提到您获得的差异。是偶然的角色是不同的还是所有传入的数据都是乱码? 请注意,由于 SerialPort.Encoding 属性。此设置会影响传入文本的解释,因为它是ASCII,Unicode,UTF8中的文本或Windows用于“原始字节”到“可读文本”转换的任何其他编码方案。

如果你正在读取一个字节数组(例如:SerialPort.Read),你应该得到你在PortMon上看到的字节。

如果您要转换为字符(SerialPort.ReadLine或SerialPort.ReadChar),那么将使用当前编码(SerialPort.Encoding属性)对数据进行编码,这解释了您所看到的差异。

如果要查看与线路上的字节具有相同二进制值的字符,则使用的良好编码是Latin-1,如这篇文章

示例:

SerialPort.Encoding = Encoding.GetEncoding("Latin1")
许可以下: CC-BY-SA归因
不隶属于 StackOverflow
scroll top