MSComm控件进行串口编程的基本步骤(转载)

时间:2022-06-18 06:07:58

MSComm控件进行串口编程的基本步骤  

2011-08-29 14:47:03|  分类: 串口编程|举报|字号 订阅

 
 

Visual C++为我们提供了一种好用的ActiveX控件Microsoft Communications Control(即MSComm)来支持应用程序对串口的访问,在应用程序中插入MSComm控件后就可以较为方便地实现对通过计算机串口收发数据。 

  要使用ActiveX控件MSComm,程序员必须将其添加入工程,其方法是: 

  (1)单击主菜单project的子菜单Add To project的Components and Controls选项; 

  (2)在弹出的"Components and Controls Gallery"对话框中选择Registered ActiveX Controls文件夹中的"Microsoft Communications Control,version 6.0"选项 

  单击其中的"Insert"按钮,MSComm控件就被增加到工程中了。与此同时,类CMSComm的相关文件mscomm.h和mscomm.cpp也一并被加入Project的Header Files和Source Files中。当然,程序员可以自己修改文件名 

1在建立的工程中插入MSComm控件 
2添加MSComm控件ID的控制变量或对象 
3对串口进行初始化,设置MSComm控件的属性 
4添加串口事件的消息处理函数OnComm()函数,在函数中根据应用需要,编写数据处理代码 
5编写串口发送等其他代码 
6关闭串口

 

一个MSComm控件只能对应一个串口,如果应用程序需要访问多个串口,必须使用多个MSComm控件。 

处理通信问题的方式:事件驱动方式(与中断原理类似,触发一个事件处理函数),查询方式。 

MSComm控件的重要属性中的一部分 
CommPort设置并返回通信端口号 
Settings以字符串的形式设置并返回波特率、奇偶校验、数据位、停止位。 
PortOpen设置并返回通信端口的状态。可以用来打开和关闭端口 
Input从接受缓冲区返回和删除字符串 
Output向传输缓冲区写一个字符串 
每个属性都通过Get/Set函数对来获取或设置控件的属性

 

属性简介 

CommPort  
设置并返回端口号 
void CMSComm::SetCommPort(short nNewValue) 
short CMSComm::GetCommPort() 

RThreshold  
在MSComm控件设置CommEvent属性为comEvReceive并产生OnComm之前,设置并返回要接收的字符数。达到设定值则触发OnComm。 
void CMSComm::SetRThreshold(short nNewValue) 
short CMSComm::GetRThreshold()

 

CTSHolding 
确定是否可通过查询CTS线的状态发送数据。硬件握手的手工查询 
void CMSComm::SetCTSHolding(BOOL bNewValue) 
BOOL CMSComm::GetCTSHolding() 

SThreshold  
在MSComm控件设置CommEvent属性为comEvSend并产生OnComm之前,设置并返回传输缓冲区中允许的最小字符数。 
void CMSComm::SetSThreshold(short nNewValue) 
short CMSComm::GetSTreshold() 

Handshaking 
设置或返回硬件握手状态,使用内部通信协议 
void CMSComm::SetHandshaking(long nNewValue) 
long CMSComm::GetHandshaking() 

InputMode 
设置或返回传输数据的类型 
void CMSComm::SetInputMode(long nNewValue) 
long CMSComm::GetInputMode()

 

DSRHolding 
确定DSR线的状态。 
void CMSComm::SetDSRHolding(BOOL bNewValue) 
BOOL CMSComm::GetDSRHolding() 

Settings 
设置并返回波特率、奇偶校验、数据位、停止位参数 
void CMSComm::SetSettings(LPCTSTR lpszNewValue) 
CString CMSComm::GetSettings() 
"BBBB,P,D,S"BBBB为波特率,P为奇偶校验,D为数据位数,S为停止位数 
缺省值为"9600,N,8,1" 

InputLen 
设置并返回Input属性从接收缓冲区读取的字符数 
void CMSComm::SetInputLen(short nNewValue) 
short CMSComm::GetInputLen() 
缺省值为0,使用Input将使MSComm控件读取接收缓冲区中的全部的内容。在使用Input前,可检查InBufferCount属性来确定缓冲区中是否已有需要数目的字符。此属性在从输出格式为定长数据的机器读取数据时非常有用。 

InBufferSize 
设置或返回输入缓冲区的大小 
void CMSComm::SetInBufferSize(short nNewValue) 
short CMSComm::GetInBufferSize() 
默认为1024Byte 

InBufferCount 
设置或返回输入缓冲区内等待读取的字节个数 
void CMSComm::SetInBufferCount(short nNewValue) 
short CMSComm::GetInBufferCount() 
当设置InBufferCount属性的值为0时,可以清空接收缓冲区(很有用的功能) 

OutBufferSize 
设置或返回发送缓冲区的大小 
void CMSComm::SetOutBufferSize(short nNewValue) 
short CMSComm::GetOutBufferSize() 
默认为512字节。此值不能太小,否则缓冲区易溢出,但太大会不必要地占用内存。 

OutBufferCount 
返回发送缓冲区的字节数或者清空发送缓冲区 
void CMSComm::SetOutBufferCount(short nNewValue) 
short CMSComm::GetOutBufferCount() 
设置值为0时清空发送缓冲区

 

InPut 
从接收缓冲区内读出数据 
VARIANT CMSComm::GetInPut() 
返回为VARIANT型变量 
当InPutMode为0时,检取数据为文本方式,变量中含有String型数据类型,当InPutMode为1时,检取数据为二进制方式,变量中含Byte数组型数据。 

OutPut 
向发送缓冲区写数据,或返回发送缓冲区当前的数据 
void CMSComm::SetOutPut(const VARIANT&newValue) 
VARIANT CMSComm::GetOutPut() 

PortOpen 
用于打开或关闭串口,或返回串口的开、关状态 
void CMSComm::SetPortOpen(BOOL bNewValue) 
BOOL CMSComm::GetPortOpen() 
TURE打开串口,FALSE关闭串口,程序终止后,MSComm控件自动关闭串口 

EOFEnable 
确定在输入过程中MSComm控件是否寻找文件结尾EOF字符,如果找到EOF字符,将停止输入并激活OnComm事件,此时CommEvent属性设置为ComEvEOF 
void CMSComm::SetEOFEnable(BOOL bNewValue) 
BOOL CMSComm::GetEOFEnable() 

DTREnalbe 
设置或返回DTR线状态。 
void CMSComm::SetDTREnable(BOOL bNewValue) 
BOOL CMSComm::GetDTREnable() 

RTSEnable 
确定是否使RTS线有效 
void CMSComm::SetRTSEnable(BOOL bNewValue) 
BOOL CMSComm::GetRTSEnable() 

Break 
设置或清除中断信号的状态 
void CMSComm::SetBreak(BOOL bNewValue) 
BOOL CMSComm::GetBreak() 

CommID 
返回一个说明通信设备的句柄。 
void CMSComm::SetCommID(long nNewValue) 
long CMSComm::GetCommID() 
该值与WindowsAPICreateFile函数返回的值一致。 

NullDiscard 
确定Null字符是否从端口传送到接收缓冲区 
void CMSComm::SetNullDiscard(BOOL bNewValue) 
BOOL CMSComm::GetNullDiscard() 

OnComm事件和CommEvent属性 
设置或返回无论何时当CommEvent属性的值变化时,就产生OnComm事件,标志发生了一个通信事件或一个错误。CommEvent属性值反映错误或者事件类型,通常在程序中的事件消息处理函数中对CommEvent事件进行处理。 
void CMSComm::SetCommEvent(short nNewValue) 
short CMSComm::GetCommEvent() 
通常由函数GetCommEvent()得到当前事件值,再进行相应的处理。

 

MSComm控件的调出及其主要属性如下。  

(1)MSComm控件的调出 

VB6.0的MSComm控件并不会主动出现在工具箱中,当需要它时,让它出现在工具箱中的步骤如下:  

  ①选择菜单上的“工程”; 

  ②在“工程”菜单中选择“部件”; 

  ③出现对话框后,在可勾选的项目中勾选Microsoft Comm Contro l6.0;

  ④按下“确定”按钮,即可在工具箱中见到“电话盒”的图标,就可以将此控件加载进行串行通信。  

(2)MSComm属性 

MSComm属性很多,其重要的属性说明如下:  

  ①CommPort:设定或返回通信端口号。端口号由1开始往上递增,最大值是16。   

      

  ②Settings:设定通信端口初始化参数。其格式为“Baud,P,D,S”,其中Baud为波特率, 可设为1200、2400、9600、14400、19200和28800等几种;P为校验位(E表示偶校验、O表示奇校验、M表示符号校验(即在校验位放置一个1的位)、S表示空白校验(即在校验位放置一个0的位)、N表示无校验位(默认值));D为数据位数,可选值为4、5、6、7、8(默认值);S表示停止位数,可选值为1、1.5、2。  

  ③PortOpen:设定并返回通信端口的状态。 

  使用串行端口之前必须先打开该端口(PortOpen=True),而在使用完毕后必须关闭该端口(PortOpen=False)。  

  ④Input:从输入缓冲区返回并清除字符。这是一种FIFO(FirstinFirstOut)机制。如Buffer$=MSComm1.Input,表示将输入缓冲区的字符读入Buffer字符串变量中。  

  ⑤Output:将一个字符或字符串写入传输缓冲区,如MSComm1.Output=“ABCD”。此即将ABCD4个字符通过串行端口传送出去。  

  ⑥InBufferCount:传回在接收缓冲区中的字符数,是指已接收,并在接收缓冲区等待读取的字符数。  

  ⑦InputMode:设定和返回类型。该属性设为0时,数据通过Input属性以文本方式取回,如设为1,则数据通过Input属性以二进制方式取回。

 

VC基于MSCOMM控件串口通讯 
在mfc中进行串口通讯最简单的方法莫过于在对话框中使用MSCOMM控件了,MSComm通信控件提供了一系列标准通信命令的接口,它允许建立串口连接,可以连接到其他通信设备(如Modem). 
还可以发送命令、进行数据交换以及监视和响应在通信过程中可能发生的各种错误和事件,从而可以用它创建全双工 、事件驱动的、高效实用的通信程序。一、用MSComm控件通信 
1.串口通信基础知识 
    一般悦来,计算机都有一个或多个串行端口,它们依次为com1、Com2、…,这些串口还提供了外部设备与pC进行数据传输和 
皿信的通道。这些串口在CPU和外设之间充当解释器的角色。当字符数据从CPU发送给外设时,这些字符数据将被转换成串行比特 
流数据;当接收数据时,比特流数据被转换为字符数据传递给CPU,再进一步说,在操作系统方面,Windows用通信驱动程序 
(COMM.DRV)调用API函数发送和接收数据,当用通信控件或声明调用API函数时,它门由COMM. DRV解释并传递给设备驱动程序, 
作为一个vB程序员,要编写通信程序.只需知道通信控件提供给Windows通信AP1函数的接口即可.换句话说,只需设定和监视通 
信控件的属性和事件即可。 
2.使用Mscomm控件 
在开始使用MSComm控件之前。需要先了解其属性、事件或错误 
属性            描述 
CommPort    设置或返回通信端口号 
Settings    以字符串的形式设置或返回波特率、奇偶校验、数据位和停止位 
PortOpen    设置或返回通信端口的状态。也可以打开和关闭端口 
Input       返回和删除接收缓冲区中的字符 
Output      将字符串写入发送缓冲区 

CommEvent属性为通信事件或错误返回下列值之一。在该控件的对象库中也可以找到这些常量。 
常量           值        描述 
ComEventBreak   1001    收到了断开信号 
ComEventCTSTO   1002    Clear To Send Timeout。在发送字符时,在系统指定的事1件内,CTS(Clear To Send)线是低电平 
ComEventDSRTO   1003    Data Set Ready Timeout。在发送字符时,在系统指定的事件内,DSR(Data Set Ready)线是低电平 
ComEventFrame   1004    数据帧错误。硬件检测到一个数据帧错误 
ComEventOverrun 1006    端口溢出。硬件中的字符尚未读,下一个字符又到达,并且丢失 
ComEventCDTO    1007    Carrier Detect Time。在发送字符时,在系统指定的事件内,CD(Carrier Detect)线是低电平。CD 
                        也称为RLSD(Receive Line Singal Detect,接收线信号检测) 
ComEventRxOver  1008    接收缓冲区溢出。在接收缓冲区中没有空间 
ComEventRxParity 1009   奇偶校验错。硬件检测到奇偶校验错误7 
ComEventTxFull  1010    发送缓冲区满。在对发送字符排队时,发送缓冲区满 

ComEventDCB     1011    检取端口DCB(Device Control Blick)时发生了没有预料到的错误 

通信事件包含了下面的设置: 
常量         值        描述 
ComEvSend      1    发送缓冲区中的字符数比Sthreshold值低 
ComEvReceive   2    接收到了Rthreshold个字符。持续产生该事件,直到使用了Input属性删除了接收缓冲区中的数据 
ComEvCTS       3    CTS(Clear To Send)线改变 
ComEvDSR       4    DSR(Data Set Ready)线改变。当DSR从1到0改变时,该事件发生 
ComEvCD        5    CD(Carrier Detect)线改变ComEvRing6检测到响铃信号。一些URAT(Universal AsynchronousReciver- 
                    -Transmitters,通用异步收发器)不支持该事件 
ComEvEOF       7    收到了EOF字符(ASCII字符26) 

Error消息(MSComm控件)下表列出了MSComm控件可捕获的错误消息: 
常量                       值      描述 
ComInvalidPropertyValue    380   无效的属性值 
ComSetNotSupported         383   属性只读 
ComGetNotSupported         394   属性只读 
ComPortOpen               8000   端口打开时该存在无效 
                          8001   超时设置必须比0值大 
ComPortInvalid            8002   无效的端口号 
                          8003   属性只在运行时有效 
                          8004   属性在运行时是只读的 
ComPortAleadyOpen         8005   端口已经打开 
                          8006   设备标识符无效或不支持 
                          8007   不支持设备的波特率 


                 8008   指定的字节大小无效 
                          8009   缺省参数错误 
                          8010   硬件不可用(被其他设备锁住) 
                          8011   函数不能分配队列 
ComNoOpen                 8012   设备没有打开 
                          8013   设备已经打开 
                          8014   不能使用通信通知 
ComSetCommStateFailed     8015   不能设置通信状态 
                          8016   不能设置通信事件屏蔽 
ComPortNotOpen            8018   该存在只在端口打开是有效 
                          8019   设备忙 
ComReadError              8020   通信设备读错误 
ComDCBError               8021   检取端口设备控制块时出现内部错误 
注意在使用的时候一定要保证两个通讯串口的设置是相同的,否则受到的信息将会产生错误!由于取值位数的不同,有可能发送的信息要读很多次才能组合成需要的信息!1。建立mfc工程,都会撒。   
   将控件加进来:打开“Project->Add To Project->Components and Controls->Registered Activex Controls”,然后选择控件:Microsoft Communication Control,version 6.0插入到当前的工程中。这样就将类 CMSComm 的相关文件 mscomm.cpp 和 mscomm.h 一并加入到了工程中。编程时只需将控件对话中的 MSComm 控件拖至你的应用对话框中就OK了 

2。定义串口对象: 
    CMSComm      m_MSComm; 

3。串口初始化: 
    
DWORD style=WS_VISIBLE; 

m_MSComm.Create(NULL,style,CRect(0,0,0,0),this,IDC_MSCOMM); 
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); 
 AfxMessageBox("Open The Serial Port 1 Failurre!"); 



4。串口数据读写: 

  MSComm 类的读写函数比较简单:GetInput()和SetOutput()。函数原形分别为VARIANT GetInput()和void SetOutput(const VARIANT newValue),均使用VARIANT类型。但PC机发送和接收数据时习惯用字符串形式。MSDN中查阅VARIANT类型,可以用BSTR表示字符串,但所有的BSTR都包含宽字符,而只有Windows NT支持宽字符,Windows 9X并不支持。所以要完成一个适应各平台的串口应用程序必须解决这个问题。这里使用CbyteArray即可解决之。 

  

发数据:在对话框对加入 按钮 控件并给你添加消息 
   void CTest_mscommDlg::OnSend()  

 // TODO: Add your control notification handler code here 
 int i,Count; 
 CString m_SendData; 
 m_SendData="Hello!";  
 Count=m_SendData.GetLength(); 
 CByteArray m_Array; 
  
 m_Array.RemoveAll(); 
 m_Array.SetSize(Count); 
  
 for(i=0;i  m_Array.SetAt(i,m_SendData[i]); 
 m_MSComm.SetOutput(COleVariant(m_Array)); 




收数据:给串口控件添加消息  
void CTest_mscommDlg::OnOnCommMscomm()  

 VARIANT m_input; 
 char *str,*str1; 
 int k,nEvent,i; 
 CString str2,m_RcvData; 
 nEvent=m_MSComm.GetCommEvent(); 
 switch(nEvent) 
 { 
 case 2: 
  k=m_MSComm.GetInBufferCount();     //接收缓冲区的字符数目 
  if(k>0) 
  { 
   m_input=m_MSComm.GetInput(); 
   str=(char*)(unsigned char*)m_input.parray->pvData; 
  } 
  i=0; 
  str1=str; 
  while(i  { 
   i++; 
   str1++; 
  } 
  *str1='\0';                               
  str2=(const char*)str;             //清除字符串中的不必要字符 
  m_RcvData=(const char *)str; 
 } 
 //数据显示处理 

m_disp+=m_RcvData; 

UpdateData(false); 


  


  
以上程序在vc6。0编译通过,在2个pc机上测试通过。

 
 
 
 
阅读(2007)评论(0)