关于RS232串口通讯

mengqimxz 2004-08-04 03:57:02
我要做一个用RS232从下位机接受数据的程序,用VC++来实现.
大家帮帮我,我希望得到任何关于这方面的东西.用现成的代码更好.
大家一定要帮我啊,我先谢谢大家了.我不会亏待大家的.
...全文
869 10 打赏 收藏 转发到动态 举报
写回复
用AI写文章
10 条回复
切换为时间正序
请发表友善的回复…
发表回复
danielzhu 2004-08-04
  • 打赏
  • 举报
回复
哇,搂住这么多人帮,我来晚了
holyeagle 2004-08-04
  • 打赏
  • 举报
回复
方法二:在单线程中实现自定义的串口通信类

  控件简单易用,但由于必须拿到对话框中使用,在一些需要在线程中实现通信的应用场合,控件的使用显得捉襟见肘。此时,若能够按不同需要定制灵活的串口通信类将弥补控件的不足,以下将介绍如何在单线程中建立自定义的通信类。

  该通信类CSimpleComm需手动加入头文件与源文件,其基类为CObject,大致建立步骤如下:

  (1) 打开串口,获取串口资源句柄

  通信程序从CreateFile处指定串口设备及相关的操作属性。再返回一个句柄,该句柄将被用于后续的通信操作,并贯穿整个通信过程。CreateFile()函数中有几个值得注意的参数设置:串口共享方式应设为0,串口为不可共享设备;创建方式必须为OPEN_EXISTING,即打开已有的串口。对于dwFlagAndAttribute参数,对串口有意义的值是FILE_FLAG_OVERLAPPED,该标志表明串口采用异步通信模式,可进行重叠操作;若值为NULL,则为同步通信方式,在同步方式下,应用程序将始终控制程序流,直到程序结束,若遭遇通信故障等因素,将导致应用程序的永久等待,所以一般多采用异步通信。

  (2)串口设置

  串口打开后,其属性被设置为默认值,根据具体需要,通过调用GetCommState(hComm,&dcb)读取当前串口设备控制块DCB(Device Control Block)设置,修改后通过SetCommState(hComm,&dcb)将其写入。再需注意异步读写的超时控制设置, 通过COMMTIMEOUTS结构设置超时,调用SetCommTimeouts(hComm,&timeouts)将结果写入。以下是温度监控程序中串口初始化成员函数:
BOOL CSimpleComm::Open( )
{
DCB dcb;

m_hIDComDev=CreateFile( "COM2",
GENERIC_READ | GENERIC_WRITE,
0,NULL,OPEN_EXISTING,FILE_ATTRIBUTE_
NORMAL|FILE_FLAG_OVE RLAPPED, 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()来等待结束信号并设置最长等待时间,举例如下:
BOOL bReadStatus;
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(UINT nIDEvent)
{
char InputData[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时间,Win 32 区分两种不同类型的线程,一种是用户界面线程UI(User Interface Thread),它包含消息循环或消息泵,用于处理接收到的消息;另一种是工作线程(Work Thread),它没有消息循环,用于执行后台任务。用于监视串口事件的线程即为工作线程。

  多线程通信类的编写在端口的配置,连接部分与单线程通信类相同,在端口配置完毕后,最重要的是根据实际情况,建立多线程之间的同步对象,如信号灯,临界区,事件等,相关细节可参考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时间。但在具体的实时监控系统中如何协调多个线程,线程之间以何种方式实现同步也是在多线程串行通信程序实现的难点。
holyeagle 2004-08-04
  • 打赏
  • 举报
回复
Windows下串行通信

  与以往DOS下串行通信程序不同的是,Windows不提倡应用程序直接控制硬件,而是通过Windows操作系统提供的设备驱动程序来进行数据传递。串行口在Win 32中是作为文件来进行处理的,而不是直接对端口进行操作,对于串行通信,Win 32 提供了相应的文件I/O函数与通信函数,通过了解这些函数的使用,可以编制出符合不同需要的通信程序。与通信设备相关的结构有COMMCONFIG ,COMMPROP,COMMTIMEOUTS,COMSTAT,DCB,MODEMDEVCAPS,MODEMSETTINGS共7个,与通信有关的Windows API函数共有26个,详细说明可参考MSDN帮助文件。以下将结合实例,给出实现串行通信的三种方法。

  实现串行通信的三种方法

  方法一:使用VC++提供的串行通信控件MSComm 首先,在对话框中创建通信控件,若Control工具栏中缺少该控件,可通过菜单Project --> Add to Project --> Components and Control插入即可,再将该控件从工具箱中拉到对话框中。此时,你只需要关心控件提供的对 Windows 通讯驱动程序的 API 函数的接口。换句话说,只需要设置和监视MSComm控件的属性和事件。

  在ClassWizard中为新创建的通信控件定义成员对象(CMSComm m_Serial),通过该对象便可以对串口属性进行设置,MSComm 控件共有27个属性,这里只介绍其中几个常用属性:

  CommPort 设置并返回通讯端口号,缺省为COM1。

  Settings 以字符串的形式设置并返回波特率、奇偶校验、数据位、停止位。

  PortOpen 设置并返回通讯端口的状态,也可以打开和关闭端口。

  Input 从接收缓冲区返回和删除字符。

  Output 向发送缓冲区写一个字符串。

  InputLen 设置每次Input读入的字符个数,缺省值为0,表明读取接收缓冲 区中的全部内容。

  InBufferCount 返回接收缓冲区中已接收到的字符数,将其置0可以清除接收缓 冲区。

  InputMode 定义Input属性获取数据的方式(为0:文本方式;为1:二进制方式)。

  RThreshold 和 SThreshold 属性,表示在 OnComm 事件发生之前,接收缓冲区或发送缓冲区中可以接收的字符数。

  以下是通过设置控件属性对串口进行初始化的实例:
BOOL CSampleDlg:: PortOpen()
{
BOOL m_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); // 打开指定的串口
return m_Opened;
}

  打开所需串口后,需要考虑串口通信的时机。在接收或发送数据过程中,可能需要监视并响应一些事件和错误,所以事件驱动是处理串行端口交互作用的一种非常有效的方法。使用 OnComm 事件和 CommEvent 属性捕捉并检查通讯事件和错误的值。发生通讯事件或错误时,将触发 OnComm 事件,CommEvent 属性的值将被改变,应用程序检查 CommEvent 属性值并作出相应的反应。在程序中用ClassWizard为CMSComm控件添加OnComm消息处理函数:
void CSampleDlg::OnComm()
{
......
switch(m_Serial.GetCommEvent())
{
case 2:
// 串行口数据接收,处理;
}
}

holyeagle 2004-08-04
  • 打赏
  • 举报
回复
转载
Visaul C++6.0中使用串行通信控件

  目前,在用计算机进行数据传输时,常用的是串行通信方式。在Visual C++的编程中,既可以用Windows API函数进行串行通信编程,也可以利用环境自带的控件进行编程。利用 API函数编写实际应用程序时,往往要考虑多线程的问题,这样编出来的程序不但十分庞大,而且结构比较复杂,继承性差,维护困难。但是使用串行通信控件就相对简单一些,而且功能强大,性能安全可靠。

  本文以Microsoft Communications Control(version 6.0)为例,介绍串行通信控件在VC++中的使用方法。

  嵌入通信控件

  使用串行通信控件前,首先要进行系统注册。

  1、在建立新工程时,将ActiveX Controls复选框选中,表示本程序支持ActiveX 控件。

  2、单击Project菜单,从中选择Add to Project,在弹出的对话框中单击Components and Controls选项,然后在弹出的对话框中再单击Registered ActiveX Controls选项。

  3、在列表中选择Microsoft Communications Control (version6.0),形状为带有红色底座的黄色电话机。单击Insert按钮,确认后,会出现一个Confirm Classes对话框,其中列出了该控件的类名(CMSComm)、头文件名(MSComm.h)、执行文件名(MSComm.cpp)。

  关闭此对话框,在程序主界面的控件列表中,可以看到该通信控件已被加入,拖动该控件就可以将其放入程序中。

  编写通信程序

  通信控件的工作原理类似于中断方式,当有通信事件发生时,如发送、接收数据,就会触发OnComm事件。在OnComm事件的处理函数中调用 GetCommEvent()函数,由该函数的返回值即可确定是什么类型的事件,并由此做出相应的处理。

  下面以接收处理串口的GPS信号为例,说明该控件在编程中的具体应用。

  利用MFC生成一个基于对话框的应用程序,对话框中的控件及其相应的消息与响应函数如下表所示。
控件名称 控件 ID 对应变量
通信控件 IDC_MSCOMM m_Comm
接收按钮 IDC_BUTTON_RECEIVEGPSDATA OnbuttonStartreceivegps()
GPS信号显示 IDC_EDIT_COMDATA m_comdata
停止接收按钮 IDC_BUTTON_ENDRECEIVEGPSDATA OnbuttonEndreceivegps()

  设置控件各部分属性:

   CommPort:1;
   DTREnable:1;
   InBufferSize:4096;
   EOFEnable:TRUE;
   Handshaking:nohandshaking;
   InputMode:1-Binary;
   NullDiscard:False;
   Rthreshold:250;
   RTSEnable:False;
   Settings:9600,n,8,1;
   Sthreshold:0。

  部分程序如下:
//开始接收GPS信号
void CGpsDlg::OnButtonStartreceivegps()
{
 //判断串口的状态,如果是关闭状态,则打开
 if(!m_Comm.GetPortOpen() )  
 //打开串口
  m_Comm.SetPortOpen(TRUE);
 UpdateData(TRUE);
}
//停止接收GPS的信号
void CGpsDlg::OnButtonEndreceivegps();
{
 //判断串口的状态,如果是打开状态,则关闭
 if( m_Comm.GetPortOpen() )
  m_Comm.SetPortOpen(FALSE);
 UpdateData(FALSE);
}
//相应的OnComm事件处理
void CGpsDlg::OnComm()
{
 m_comdata.Empty();
 VARIANT m_input1;
 COleSafeArray m_input2;
 LONG length,i;
 BYTE data[1024];
 CString str;
 //接收缓冲区内字符
 if(m_Comm.GetCommEvent()== 2)
 {
  m_comdata.Empty();
  //读取缓冲区内的数据
  m_input1=m_Comm.GetInput();
  //将VARIANT型变量转换为ColeSafeArray型变量
  m_input2=m_input1;
  //确定数据长度
  length=m_input2.GetOneDimSize();
  //将数据转换为BYTE型数组
  for(i=0;i< length;i++)
   m_input2.GetElement(&i,data+i);
   //将数组转换为CString型变量
  for(i=0;i< length;i++) {
   char a=* (char *)(data+i);
   str.Format(“%c",a);
   m_comdata += str;
  }
  /*在上面的程序中,先将VARIANT型变量
  转换为ColeSafeArray型变量,再将其转换为
  BYTE型数组,然后将数组转换为CString型变量,
  以满足不同的变量类型显示数据的需要。*/
  //提取GPS信号
  CString m_zjz;
  m_zjz=m_comdata;
  //寻找GPS信号的头标志
  int s;
  s=m_zjz.Find(“$GPRMC,");
  CString m_gps;
  //NUM为所提取GPS信号的长度
  m_gps=m_zjz.Mid(s,NUM);
  int x;
  x=m_gps.GetLength();
  //信号长度不符合要求,则拒绝接收
  if (x != NUM) {
   m_gps.Empty();
  }
  //寻找所需要的GPS信号的头标志
  int y;
  y=m_gps.Find(“$GPRMC,");
  if (y != 0)
  {
   m_gps.Empty();
  }
  //m-gpsdata是已声明的类成员变量
  m_gpsdata=m_gps;
 }
}




mengqimxz 2004-08-04
  • 打赏
  • 举报
回复
用QQ发好了 18117310这样会快一点,谢谢你
Melly 2004-08-04
  • 打赏
  • 举报
回复
BYTE Inport(WORD p1)
{
BYTE temp;
_asm
{
mov dx,p1
in al,dx
mov temp,al
}
return temp;
}

void Outport(WORD p1,BYTE num)
{
_asm
{
mov dx,p1
mov al,num
out dx,al
}
}
mengqimxz 2004-08-04
  • 打赏
  • 举报
回复
liu_pingmxz@126.com
谢谢.
DentistryDoctor 2004-08-04
  • 打赏
  • 举报
回复
给个E-Mail,给你发个基于读写线程的OVERLAP的代码。
pmwin 2004-08-04
  • 打赏
  • 举报
回复
呵呵,给你解释楼上的各个函数吧
打开串口,设置串口参数,写串口,读串口,关闭串口。
代码很多的,串口操作也有很多种方式,查找一下就好
niu_a 2004-08-04
  • 打赏
  • 举报
回复
createfile
SetCommState
readfile
writefile
CloseHandle

16,471

社区成员

发帖
与我相关
我的任务
社区描述
VC/MFC相关问题讨论
社区管理员
  • 基础类社区
  • Web++
  • encoderlee
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

        VC/MFC社区版块或许是CSDN最“古老”的版块了,记忆之中,与CSDN的年龄几乎差不多。随着时间的推移,MFC技术渐渐的偏离了开发主流,若干年之后的今天,当我们面对着微软的这个经典之笔,内心充满着敬意,那些曾经的记忆,可以说代表着二十年前曾经的辉煌……
        向经典致敬,或许是老一代程序员内心里面难以释怀的感受。互联网大行其道的今天,我们期待着MFC技术能够恢复其曾经的辉煌,或许这个期待会永远成为一种“梦想”,或许一切皆有可能……
        我们希望这个版块可以很好的适配Web时代,期待更好的互联网技术能够使得MFC技术框架得以重现活力,……

试试用AI创作助手写篇文章吧