我创建了两个函数来读取和写入串行端口,我使用visual studio 2012,Windows 7,64位操作系统和使用RS-232串行线在c ++中编码。我正在连接的电路板应该发送5个字符,TRG 1,按下按钮,代码工作,但输出并不总是正确的值。
char serialRead()
{
char input[5];
DCB dcBus;
HANDLE hSerial;
DWORD bytesRead, eventMask;
COMMTIMEOUTS timeouts;
hSerial = CreateFile (L"\\\\.\\COM13", GENERIC_READ, 0, NULL, OPEN_EXISTING, 0, NULL);
if (hSerial == INVALID_HANDLE_VALUE)
{
cout << "error opening handle\n";
}
else
{
cout << "port opened\n";
}
dcBus.DCBlength = sizeof(dcBus);
if ((GetCommState(hSerial, &dcBus) == 0))
{
cout << "error getting comm state\n";
}
dcBus.BaudRate = CBR_9600;
dcBus.ByteSize = DATABITS_8;
dcBus.Parity = NOPARITY;
dcBus.StopBits = ONESTOPBIT;
if ((GetCommState(hSerial, &dcBus) == 0))
{
cout << "error setting comm state\n";
}
if ((GetCommTimeouts(hSerial, &timeouts) == 0))
{
cout << "error getting timeouts\n";
}
timeouts.ReadIntervalTimeout = 10;
timeouts.ReadTotalTimeoutMultiplier = 1;
timeouts.ReadTotalTimeoutConstant = 500;
timeouts.WriteTotalTimeoutMultiplier = 1;
timeouts.WriteTotalTimeoutConstant = 500;
if (SetCommTimeouts(hSerial, &timeouts) == 0)
{
cout << "error setting timeouts\n";
}
if (SetCommMask(hSerial, EV_RXCHAR) == 0)
{
cout << "error setting comm mask\n";
}
if (WaitCommEvent(hSerial, &eventMask, NULL))
{
if (ReadFile(hSerial, &input, 5, &bytesRead, NULL) !=0)
{
for (int i = 0; i < sizeof(input); i++)
{
cout << input[i];
}
cout << endl;
}
else
{
cout << "error reading file\n";
}
}
else
{
cout << "error waiting for comm event\n";
}
switch (input[4])
{
case '1' :
CloseHandle(hSerial);
return '1';
break;
case '2' :
CloseHandle(hSerial);
return '2';
break;
case '3' :
CloseHandle(hSerial);
return '3';
break;
case '4' :
CloseHandle(hSerial);
return '4';
break;
case '5':
CloseHandle(hSerial);
return '5';
break;
default :
CloseHandle(hSerial);
return '9';
break;
}
}
代码运行成功,因为端口配置正确并且正在传输数据。输出变化,大多数时候输出将打印整个“TRG 1”,但随机(看起来),输出将是“TRG |} |}”或“T |} |} |} |}”,即它将成为字符串的一部分,每个缺失的字符将被替换为“|}”而不是正确的字符。这是一个问题,因为我希望能够为触发器发送不同的值并运行该变量的切换。
我对串行通信比较陌生,而不是专家程序员,所以我想知道发生了什么事?
答案 0 :(得分:4)
串行通信不是基于数据包的。这些信息不会出现在整个消息必须一次读取的包中;相反,它是一个流,所以你可以阅读半个消息,一个完整的消息,一个以上的消息等。
正如zdan在评论中所说,您需要检查从ReadFile
读取的字节数,并使用它来编写作为您的消息的5个字符的包。
具体来说,只有读取的返回字节数的前几个字符才有效;其余的都是垃圾。