VC中如何从串口读取数据.docx
《VC中如何从串口读取数据.docx》由会员分享,可在线阅读,更多相关《VC中如何从串口读取数据.docx(10页珍藏版)》请在冰豆网上搜索。
![VC中如何从串口读取数据.docx](https://file1.bdocx.com/fileroot1/2022-11/25/4b7caa91-7144-42b8-bd8f-39782385d89f/4b7caa91-7144-42b8-bd8f-39782385d89f1.gif)
VC中如何从串口读取数据
方法一:
使用VC++提供的串行通信控件MSComm首先,在对话框中创建通信控件,若Control工具栏中缺少该控件,可通过菜单Project–>AddtoProject–>ComponentsandControl插入即可,再将该控件从工具箱中拉到对话框中。
此时,你只需要关心控件提供的对Windows通讯驱动程序的API函数的接口。
换句话说,只需要设置和监视MSComm控件的属性和事件。
在ClassWizard中为新创建的通信控件定义成员对象(CMSCommm_Serial),通过该对象便可以对串口属性进行设置,MSComm控件共有27个属性,这里只介绍其中几个常用属性:
CommPort设置并返回通讯端口号,缺省为COM1。
Settings以字符串的形式设置并返回波特率、奇偶校验、数据位、停止位。
PortOpen设置并返回通讯端口的状态,也可以打开和关闭端口。
Input从接收缓冲区返回和删除字符。
Output向发送缓冲区写一个字符串。
InputLen设置每次Input读入的字符个数,缺省值为0,表明读取接收缓冲区中的全部内容。
InBufferCount返回接收缓冲区中已接收到的字符数,将其置0可以清除接收缓冲区。
InputMode定义Input属性获取数据的方式(为0:
文本方式;为1:
二进制方式)。
RThreshold和SThreshold属性,表示在OnComm事件发生之前,接收缓冲区或发送缓冲区中可以接收的字符数。
以下是通过设置控件属性对串口进行初始化的实例:
BOOLCSampleDlg:
:
PortOpen()
{
BOOLm_Opened;
……
m_Serial.SetCommPort
(2);//指定串口号
m_Serial.SetSettings(“4800,N,8,1″);//通信参数设置
m_Serial.SetInBufferSize(1024);//指定接收缓冲区大小
m_Serial.SetInBufferCount(0);//清空接收缓冲区
m_Serial.InputMode
(1);//设置数据获取方式
m_Serial.SetInputLen(0);//设置读取方式
m_Opened=m_Serail.SetPortOpen
(1);//打开指定的串口
returnm_Opened;
}
打开所需串口后,需要考虑串口通信的时机。
在接收或发送数据过程中,可能需要监视并响应一些事件和错误,所以事件驱动是处理串行端口交互作用的一种非常有效的方法。
使用OnComm事件和CommEvent属性捕捉并检查通讯事件和错误的值。
发生通讯事件或错误时,将触发OnComm事件,CommEvent属性的值将被改变,应用程序检查CommEvent属性值并作出相应的反应。
在程序中用ClassWizard为CMSComm控件添加OnComm消息处理函数:
voidCSampleDlg:
:
OnComm()
{
……
switch(m_Serial.GetCommEvent())
{
case2:
//串行口数据接收,处理;
}
}
方法二:
在单线程中实现自定义的串口通信类
控件简单易用,但由于必须拿到对话框中使用,在一些需要在线程中实现通信的应用场合,控件的使用显得捉襟见肘。
此时,若能够按不同需要定制灵活的串口通信类将弥补控件的不足,以下将介绍如何在单线程中建立自定义的通信类。
该通信类CSimpleComm需手动加入头文件与源文件,其基类为CObject,大致建立步骤如下:
(1)打开串口,获取串口资源句柄
通信程序从CreateFile处指定串口设备及相关的操作属性。
再返回一个句柄,该句柄将被用于后续的通信操作,并贯穿整个通信过程。
CreateFile()函数中有几个值得注意的参数设置:
串口共享方式应设为0,串口为不可共享设备;创建方式必须为OPEN_EXISTING,即打开已有的串口。
对于dwFlagAndAttribute参数,对串口有意义的值是FILE_FLAG_OVERLAPPED,该标志表明串口采用异步通信模式,可进行重叠操作;若值为NULL,则为同步通信方式,在同步方式下,应用程序将始终控制程序流,直到程序结束,若遭遇通信故障等因素,将导致应用程序的永久等待,所以一般多采用异步通信。
(2)串口设置
串口打开后,其属性被设置为默认值,根据具体需要,通过调用GetCommState(hComm,&dcb)读取当前串口设备控制块DCB(DeviceControlBlock)设置,修改后通过SetCommState(hComm,&dcb)将其写入。
再需注意异步读写的超时控制设置,通过COMMTIMEOUTS结构设置超时,调用SetCommTimeouts(hComm,&timeouts)将结果写入。
以下是温度监控程序中串口初始化成员函数:
BOOLCSimpleComm:
:
Open()
{
DCBdcb;
m_hIDComDev=CreateFile(“COM2″,
GENERIC_READ|GENERIC_WRITE,
0,NULL,OPEN_EXISTING,FILE_ATTRIBUTE_
NORMAL|FILE_FLAG_OVERLAPPED,NULL);
//打开串口,异步操作
if(m_hIDComDev==NULL)return(FALSE);
dcb.DCBlength=sizeof(DCB);
GetCommState(m_hIDComDev,&dcb);//获得端口默认设置
dcb.BaudRate=CBR_4800;
dcb.ByteSize=8;
dcb.Parity=NOPARITY;
dcb.StopBits=(BYTE)ONESTOPBIT;
……}
(3)串口读写操作
主要运用ReadFile()与WriteFile()API函数,若为异步通信方式,两函数中最后一个参数为指向OVERLAPPED结构的非空指针,在读写函数返回值为FALSE的情况下,调用GetLastError()函数,返回值为ERROR_IO_PENDING,表明I/O操作悬挂,即操作转入后台继续执行。
此时,可以用WaitForSingleObject()来等待结束信号并设置最长等待时间,举例如下:
BOOLbReadStatus;
bReadStatus=ReadFile(m_hIDComDev,buffer,
dwBytesRead,&dwBytesRead,&m_OverlappedRead);
if(!
bReadStatus)
{
if(GetLastError()==ERROR_IO_PENDING)
{
WaitForSingleObject(m_OverlappedRead.hEvent,1000);
return((int)dwBytesRead);
}
return(0);
}
return((int)dwBytesRead);
定义全局变量m_Serial作为新建通信类CSimpleComm的对象,通过调用类的成员函数即可实现所需串行通信功能。
与方法一相比,方法二赋予串行通信程序设计较大的灵活性,端口的读写可选择较简单的查询式,或通过设置与外设数据发送时间间隔TimeCycle相同的定时器:
SetTimer(1,TimeCycle,NULL),进行定时读取或发送。
CSampleView:
:
OnTimer(UINTnIDEvent)
{
charInputData[30];
m_Serial.ReadData(InputData,30);
//数据处理
}
若对端口数据的响应时间要求较严格,可采用事件驱动I/O读写,Windows定义了9种串口通信事件,较常用的有:
EV_RXCHAR:
接收到一个字节,并放入输入缓冲区。
EV_TXEMPTY:
输出缓冲区中的最后一个字符发送出去。
EV_RXFLAG:
接收到事件字符(DCB结构中EvtChar成员),放入输入缓冲区。
在用SetCommMask()指定了有用的事件后,应用程序可调用WaitCommEvent()来等待事件的发生。
SetCommMask(hComm,0)可使WaitCommEvent()中止。
方法三多线程下实现串行通信
方法一,二适用于单线程通信。
在很多工业控制系统中,常通过扩展串口连接多个外设,各外设发送数据的重复频率不同,要求后台实时无差错捕捉,采集,处理,记录各端口数据,这就需要在自定义的串行通信类中创建端口监视线程,以便在指定的事件发生时向相关的窗口发送通知消息。
线程的基本概念可详见VC++参考书目,Windows内部的抢先调度程序在活动的线程之间分配CPU时间,Win32区分两种不同类型的线程,一种是用户界面线程UI(UserInterfaceThread),它包含消息循环或消息泵,用于处理接收到的消息;另一种是工作线程(WorkThread),它没有消息循环,用于执行后台任务。
用于监视串口事件的线程即为工作线程。
多线程通信类的编写在端口的配置,连接部分与单线程通信类相同,在端口配置完毕后,最重要的是根据实际情况,建立多线程之间的同步对象,如信号灯,临界区,事件等,相关细节可参考VC++中的同步类。
一切就绪后即可启动工作线程:
CWinThrea*CommThread=AfxBegin
Thread(CommWatchThread,//线程函数名
(LPVOID)m_pTTYInfo,//传递的参数
THREAD_PRIORITY_ABOVE_NORMAL,//设置线程优先级
(UINT)0,//最大堆栈大小
(DWORD)Create_SUSPENDED,//创建标志
(LPSECURITY_ATTRIBUTES)NULL);//安全性标志
同时,在串口事件监视线程中:
if(WaitCommEvent(pTTYInfo->idComDev,&dwEvtMask,NULL))
{
if((dwEvtMask&pTTYInfo->dwEvtMask)==pTTYInfo->dwEvtMask)
{
WaitForSingleObject(pTTYInfo->hPostEvent,0xFFFFFFFF);
ResetEvent(pTTYInfo->hPostEvent);//置同步事件对象为非信号态
:
:
PostMessage(CSampleView,ID_COM1_DATA,0,0);//发送通知消息
}
}
用PostMessage()向指定窗口的消息队列发送通知消息,相应地,需要在该窗口建立消息与成员函数间的映射,用ON_MESSAGE将消息与成员函数名关联。
BEGIN_MESSAGE_MAP(CSampleView,CView)
//{{AFX_MSG_MAP(CSampleView)
ON_MESSAGE(ID_COM1_DATA,OnProcessCom1Data)
ON_MESSAGE(ID_COM2_DATA,OnProcessCom2Data)
…..
//}}AFX_MSG_MAP
END_MESSAGE_MAP()
然后在各成员函数中完成对各串口数据的接收处理,但必须保证在下一次监测到有数据到来之前,能够完成所有的中间处理工作。
否则将造成数据的捕捉错误。
多线程的实现可以使得各端口独立,准确地实现串行通信,使串口通信具有更广泛的灵活性与严格性,且充分利用了CPU时间。
但在具体的实时监控系统中如何协调多个线程,线程之间以何种方式实现同步也是在多线程串行通信程序实现的难点。
一个读串口的函数:
HANDLEm_hIDComDev;
intReceiveComm(char*RecCommData)
{
DWORDdRead,dReadNum;
COMSTATComStat;
LPDWORDComError;
char*Data;
ClearCommBreak(m_hIDComDev);
ClearCommError(m_hIDComDev,ComError,&ComStat);
dRead=ReadFile(m_hIDComDev,Data,ComStat.cbInQue,&dReadNum,NULL);//接收200个字符
//dReadNum为实际接收字节数
PurgeComm(m_hIDComDev,PURGE_RXCLEAR);//清空接收缓冲区
for(inti=0;i{
*RecCommData=*Data;
RecCommData++;
Data++;
}
*RecCommData=‘\0′;
if(dRead)
return1;
else
return0;
}
VC不是语言,VC是编程环境,C,C++是编程语言,VC中编写C,C++是比较方便的,较好的环境.
vc编程源程序要求用C++语言来编写,在这裏写一下C++语言就可以了吧!
C++支持面向对象的程序设计方法,并可以使用MS的MFC,所开发的软件稳定性好,可移植性强,软件与硬件相互独立.C++也是一种混合型的程序设计语言,既可以支持传统的面向过程的程序设计,也支持现在的面向对象的程序设计(面向对象的方法实现了软件的重用问题,加速了软件的开发过程),C++既适合作为系统软件的描述语言也适合应用软件的开发语言,与C语言比它的错误检查机制更强,还提供了相关的检查类,以编写模块化程度高,可重用性和可维护性佳更适合於大,中型程序的开发.
VC++下用MSComm控件实现串口通讯
首先,在对话框中创建通信控件,若Control工具栏中缺少该控件,可通过菜单Project-->AddtoProject-->ComponentsandControl插入即可,再将该控件从工具箱中拉到对话框中。
此时,你只需要关心控件提供的对Windows通讯驱动的API函数的接口。
换句话说,只需要设置和监视MSComm控件的属性和事件。
打开所需串口后,需要考虑串口通信的时机。
在接收或发送数据过程中,可能需要监视并响应一些事件和错误,所以事件是处理串行端口交互作用的一种非常有效的方法。
使用OnComm事件和CommEvent属性捕捉并检查通讯事件和错误的值。
发生通讯事件或错误时,将触发OnComm事件,CommEvent属性的值将被改变,应用程序检查CommEvent属性值并作出相应的反应
//若是在SDI中使用该控件则要调用下两句,在对话框程序中该语句有MFC自己创建
//所以不用人为添加
DWORDstyle=WS_VISIBLE;
m_MSComm.Create(NULL,style,CRect(0,0,0,0),this,IDC_MSCOMM1);
//串口控件的初始化
DWORDstyle=WS_VISIBLE;
m_MSComm.Create(NULL,style,CRect(0,0,0,0),this,IDC_MSCOMM1);
if(m_MSComm.GetPortOpen())//如果串口是打开的,则行关闭串口
{
m_MSComm.SetPortOpen(FALSE);
}
m_MSComm.SetCommPort
(1);//选择COM1
m_MSComm.SetInBufferSize(1024);//接收缓冲区
m_MSComm.SetOutBufferSize(1024);//发送缓冲区
m_MSComm.SetInputLen(0);//设置当前接收区数据长度为0,表示全部读取
m_MSComm.SetInputMode
(1);//以二进制方式读写数据
m_MSComm.SetRThreshold
(1);//接收缓冲区有1个及1个以上字符时,将引发接收数据的OnComm事件
m_MSComm.SetSettings("9600,n,8,1");//波特率9600无检验位,8个数据位,1个停止位
if(!
m_MSComm.GetPortOpen())//如果串口没有打开则打开
m_MSComm.SetPortOpen(TRUE);//打开串口
else
m_MSComm.SetOutBufferCount(0);
//控件事件的响应声明
//*.h
//{{AFX_MSG(CGolfView)
afx_msgBOOLOnComm();
DECLARE_EVENTSINK_MAP()
//}}AFX_MSG
//*.cpp
BEGIN_EVENTSINK_MAP(CGolfView,CView)
//{{AFX_EVENTSINK_MAP(CAboutDlg)
ON_EVENT(CGolfView,IDC_MSCOMM1,1/*OnComm*/,OnComm,VTS_NONE)
//}}AFX_EVENTSINK_MAP
END_EVENTSINK_MAP()
//控件事件的响应
BOOLCGolfView:
:
OnComm()
{
VARIANTvariant_inp;
COleSafeArraysafearray_inp;
LONGlen,k;
BYTErxdata[2048];//设置BYTE数组An8-bitintegerthatisnotsigned.
CStringstrtemp;
switch(m_MSComm.GetCommEvent())
{
case1:
//comEvSend发送数据
break;
case2:
//comEvReceive读取数据
//MessageBox(_T("读取数据事件"),_T("TRACE"),MB_OK);
variant_inp=m_MSComm.GetInput();//读缓冲区
safearray_inp=variant_inp;//VARIANT型变量转换为ColeSafeArray型变量
len=safearray_inp.GetOneDimSize();//得到有效数据长度
//接受数据
for(k=0;k {
safearray_inp.GetElement(&k,rxdata+k);//转换为BYTE型数组
BYTEbt=*(char*)(rxdata+k);//字符型
strtemp.Format("%c",bt);//将字符送入临时变量strtemp存放
recd+=strtemp;
}
//UpdateData(TRUE);
break;
default:
//传输事件出错
m_MSComm.SetOutBufferCount(0);
break;
}
UpdateData(FALSE);//更新图象内容
returnTRUE;
}