WinCE下的串口通信开发
一、利用Visual Basic 开发很简单,因为有现成的控件可以直接调用
以VS2005为例,首先要从工具箱中找到SerialPort控件,将其拖到设计视图的下方,因为这些控件(包括定时控件Timer等)在运行的时候不会显示在视图中的。可以通过两种方法对串口进行配置,即点击控件的属性页码,在里面直接修改;也可以通过代码来配置成功,主要如下:
If VBSerial.IsOpen Then
VBSerial.Close()
End If
'串口设置:9600-N-8-1
With VBSerial
.PortName = "COM2"
.BaudRate = 9600
.Parity = Parity.None
.ReceivedBytesThreshold = 13 '接受的数据包长度
.DataBits = 8
.StopBits = StopBits.One
End With
串口的基本配置完成以后,下一步就是重要的事件属性了,既要设置数据接收的事件属性,在串口控件的属性页,点击事件标志(闪电状的),右方输入事件函数的名字即可,然后在事件函数中用Try,Catch读取串口字符,代码如下:
Dim rec_byte(LENGTH) As Byte '串口接收数据包 LENGTH是一个宏定义的数据包长度
Try
data_num = VBSerial.BytesToRead '串口接收到的字节数,用于测试调校程序
Me.VBSerial.Read(rec_byte, 0, LENGTH)
…………
…………
操作数组rec_byte(LENGTH)即可完成拆包等工作
Catch ex AsTimeoutException '超时后的处理,包括读、写超时等
Timer1.Enabled = False
VBSerial.Close()
System.Threading.Thread.Sleep(300)
End Try
至此,即可完成Vb2005 开发WinCE的串口简单操作。
二、利用Visual C++开发
Visual C++没有封装好的现成通信控件可以直接调用,故需要用API函数或者第三方的类,这里推荐一个现成的类,见参考文件1中第十章的内容,这个类的详细内容与使用方法,书中写的已经非常仔细,这里我只将我的一点建议叙述,该类的读线程代码主要如下:
while (TRUE)
{
if (WaitCommEvent(ceSeries->m_hComm,&evtMask,0))
{
SetCommMask(ceSeries->m_hComm, EV_RXCHAR | EV_CTS | EV_DSR );
//表示串口收到字符
if (evtMask & EV_RXCHAR)
{
ClearCommError(ceSeries->m_hComm,&dwReadErrors,&cmState);
if(cmState.cbInQue == (ceSeries->m_RThreshold) )//这个if内的内容是我的测试
{
willReadLen= cmState.cbInQue ;
if (willReadLen <= 0)
{
continue;
}
//分配内存
readBuf= new BYTE[willReadLen];
ZeroMemory(readBuf,willReadLen);
//读取串口数据
ReadFile(ceSeries->m_hComm,readBuf, willReadLen, &actualReadLen,0);
//如果读取的数据大于0,
if (actualReadLen>0)
{
//触发读取回调函数
if (ceSeries->m_OnSeriesRead)
{
ceSeries->m_OnSeriesRead(ceSeries->m_pOwner,readBuf,actualReadLen);
}
}
//释放内存
delete[] readBuf;
readBuf= NULL;
}
}
}
//如果收到读线程退出信号,则退出线程
if(WaitForSingleObject(ceSeries->m_hReadCloseEvent,500) == WAIT_OBJECT_0)
{
break;
}
}
原来的代码中没有那个if来限制,这样每次读取的字符数最多有8个(之前我测试的时候,发现如果数据包长度大于8,就需要分几次读完,给实际应用中造成了不便),此处我加一个if语句来判别:
if(cmState.cbInQue ==(ceSeries->m_RThreshold) )
cmState.cbInQue表示现在的接受缓冲区队列中的字符个数,而ceSeries->m_RThreshold是需要预先设置好的,在串口的初始化中设置:
if(m_pSerial->OpenPort (this ,2,9600,0,8,0,16))
SetTimer(1,1000,NULL);
else
AfxMessageBox(L"串口打开失败");
最后一个16代表设置的数据包字符个数,这个是由通信协议来确定的。这样每次当检测事件为收到字符时,先等到接收缓冲区队列中的字符个数是16,然后才一次性读取字符,再发送消息给主线程,便于在主线程中一次处理拆包解析工作。
参考文献
[1] 汪兵.Windows CE 嵌入式高级编程与实例详解(用C++实现),[M].中国水利水电出版社,2008