在串口通讯程序中,经常要收到数据包,常有网友问及如何从这些数据包中提取需要的数据,如何处理校验等,在这篇文章里我举两个例子予以说明,程序说明为VC++6.0。关于串口编程建立程序的细节,请参阅我主页上的其它文章。同时,此文也适于其它通讯
程序中艰数据报文的处理。
首先,应该指出的是,所有这些处理均在串口事件处理函数oncommunication()中进行。每当串口缓冲区中有一个或一个以上字符时触发串口通讯事件,该事件就驱动(调用)串口事件通讯处理函数oncommunication(),在这里就可以对接收到的数据进行处理,提取需要的数据。
举两个例子,一个是较为简单的位数据格式的处理,另一个是NMEA无线通讯格式的处理,最后回答一位网友提出的
问题,大家也可以探讨一下。
1.问题:
一个数据包,其串头为一个字符,字符值为7EH(16进制)''~'',其后紧跟一字符‘E’,然后是数据串,串尾也为字符值为7EH的一个字符:
即 ~Exxxxxx~ 如何处理这些数据?
我们仍以串口调试助手源
程序及其详细编程过程之一 中的OnComm()处理为例:
void CSCommTestDlg::OnComm()
{
// TODO: Add your control notification handler code here
VARIANT variant_inp;
COleSafeArray safearray_inp;
LONG len,k;
BYTE rxdata[2048]; //设置BYTE数组 An 8-bit integerthat is not signed.
CString strtemp;
if(m_ctrlComm.GetCommEvent()==2) //事件值为2表示接收缓冲区内有字符
{ ////////以下你可以根据自己的通信协议加入处理代码
variant_inp=m_ctrlComm.GetInput(); //读缓冲区
safearray_inp=variant_inp; //VARIANT型变量转换为ColeSafeArray型变量
len=safearray_inp.GetOneDimSize(); //得到有效数据长度
for(k=0;k<len;k++)
safearray_inp.GetElement(&k,rxdata+k);//转换为BYTE型数组
for(k=0;k<len;k++) //将数组转换为Cstring型变量
{
BYTE bt=*(char*)(rxdata+k); //字符型
strtemp.Format("%c",bt); //将字符送入临时变量strtemp存放
m_strRXData+=strtemp; //加入接收编辑框对应字符串,在这儿,编辑框不是必须的,可做相应处理
char ch=(char)bt;
if(ch==''E'')
{
//在此处设置一个可以接收数据的全局标志,说明接收到数据前的‘E’标志了,下一步可以读数据了,同时将m_strRXData清空
flag=2;
m_strRXData.Empty(); //下一次接收的便为有用的数据
}
if(ch==0x7e)
{
flag=1; //下面可以提取数据了
}
if(flag==1) //标志为1,
{
//提取数据
flag=0; //提取完后,置标志为0
}
}
}
//UpdateData(FALSE); //更新编辑框内容
}
2 NMEA无线通讯格式的处理
2.1 NMEA-0183报文格式
字符串(ASCII字符)格式如下:
$XXXX,XX,XX,XX,……*hh<CR><LF>
$:串头
XXXX: 串头
XX:数据字段,字母或数字
XX:数据字段,字母或数字
XX:数据字段,字母或数字
,:逗号
……
*:星号,串尾
hh:$与*之间所有字符代码的校验和,(注意:校验和h为半Byte校验,*后第1个h表示高4位校验和,第2个h表示低4位校验和。得到校验值后,再转换成ASCII字符。)
<CR>:0DH,回车控制符
<LF>:0AH,换行控制符
2.2 校验处理
由于数据是动态接收,所以数据的处理也是动态进行,尽管有时会收到几个字符才触发一个串口事件,但字符的接收是一个一个接收的,因此就可以在
程序中先判断串头$是否到达,若串头到达,就可以开始计算校验,直至串