我正在使用来自“大牌”之一的第 3 方串行端口组件(是的,我已向他们寻求支持,但存在时区差异,我需要尽快解决此问题)。该组件已经存在多年,我没有理由相信问题出在它身上(硬件同上)。
硬件规范说,如果我将某个字符串写入串行端口,以回车终止,然后读取,它将以特定格式的 8 字符串回复,再次以回车终止。
代码可以正确运行几个小时,并根据它读取的内容更新 GUI。
但是,当 GUI 上有任何用户活动时,我会从串行端口读取垃圾。
我首先注意到它是在单击导致模式表单打开然后关闭表单的按钮时。
但是,当简单地拖动 TStringGrid 的 scollbar 时,我也会看到它。
这是代码。有什么建议吗?
更新:组件是线程化的,供应商同意这里的海报-串行端口是异步设备。我已更改代码以编写数据请求并处理组件的 OnCharReceived() 事件处理程序中收到的每个字符。塔恩斯克的所有帮助。
function TForm1.ReadChannelValueFromSerialPort(
device_number : String; channel_number : String) : Real;
const SLEEP_TIME = 50; // ms
NUM_READ_ATTEMPTS = 10;
var serialPortInput : String;
read_attempt_counter : Integer;
messageString : String;
begin
WriteToSerialPort('#' + device_number + channel_number + #13);
serialPortInput := '';
read_attempt_counter := 0;
while Length(serialPortInput) = 0 do
begin
try
Application.ProcessMessages();
serialPortInput := serialPortInput + SerialPort.ReadText();
except
on E: Exception do
begin
messageString := 'Can''t read from serial port' ;
MessageDlg(messageString, mtError, [mbOK], 0);
Halt(0);
end;
end;
Inc(read_attempt_counter);
if (read_attempt_counter = NUM_READ_ATTEMPTS)
and (Length(serialPortInput) = 0) then
begin
messageString := 'Can''t read from serial port after trying ' +
IntToStr(NUM_READ_ATTEMPTS) + ' times in ' +
FloatToStr((SLEEP_TIME * NUM_READ_ATTEMPTS) / 1000) + ' seconds';
MessageDlg(messageString, mtError, [mbOK], 0);
Halt(0);
end;
if (Length(serialPortInput) = 0) then
Sleep(SLEEP_TIME);
end;
if Copy(serialPortInput, 1, 1) <> '>' then
begin
DebugBreak();
MessageDlg('Invalid value read from serial port "' +
serialPortInput + '"', mtError, [mbOK], 0);
Halt(0);
end;
// drop the3 leading >
serialPortInput := Copy(serialPortInput, 2, Length(serialPortInput) - 1);
serialPortInput := TrimRight(serialPortInput); // just in case
Result := StrToFloat(serialPortInput);
end; // ReadChannelValueFromSerialPort();