中国科学院 王颖 摘要 本文介绍了在Windows平台下串行通信的实现机制讨论了根据不同的条件用Visual C++ 设计串行通信程序的三种方法并结合实际实现对温度数据的接收监控 在实验室和工业应用中串口是常用的计算机与外部串行设备之间的数据传输通道由于串行通信方便易行所以应用广泛依据不同的条件实现对串口的灵活编程控制是我们所需要的 在光学镜片镀膜工艺中用单片机进行多路温度数据采集控制采集结果以串行方式进入主机每隔S向主机发送一次采样数据主机向单片机发送相关的控制命令实现串行数据接收处理记录显示实时绘制曲线串行通信程序开发环境为 VC++ Windows下串行通信 与以往DOS下串行通信程序不同的是Windows不提倡应用程序直接控制硬件而是通过Windows操作系统提供的设备驱动程序来进行数据传递串行口在Win 中是作为文件来进行处理的而不是直接对端口进行操作对于串行通信Win 提供了相应的文件I/O函数与通信函数通过了解这些函数的使用可以编制出符合不同需要的通信程序与通信设备相关的结构有COMMCONFIG COMMPROPCOMMTIMEOUTSCOMSTATDCBMODEMDEVCAPSMODEMSETTINGS共个与通信有关的Windows API函数共有个详细说明可参考MSDN帮助文件以下将结合实例给出实现串行通信的三种方法 实现串行通信的三种方法 方法一使用VC++提供的串行通信控件MSComm 首先在对话框中创建通信控件若Control工具栏中缺少该控件可通过菜单Project > Add to Project > Components and Control插入即可再将该控件从工具箱中拉到对话框中此时你只需要关心控件提供的对 Windows 通讯驱动程序的 API 函数的接口换句话说只需要设置和监视MSComm控件的属性和事件 在ClassWizard中为新创建的通信控件定义成员对象(CMSComm m_Serial)通过该对象便可以对串口属性进行设置MSComm 控件共有个属性这里只介绍其中几个常用属性 CommPort 设置并返回通讯端口号缺省为COM Settings 以字符串的形式设置并返回波特率奇偶校验数据位停止位 PortOpen 设置并返回通讯端口的状态也可以打开和关闭端口 Input 从接收缓沖区返回和删除字符 Output 向发送缓沖区写一个字符串 InputLen 设置每次Input读入的字符个数缺省值为表明读取接收缓沖 区中的全部内容 InBufferCount 返回接收缓沖区中已接收到的字符数将其置可以清除接收缓 沖区 InputMode 定义Input属性获取数据的方式(为文本方式为二进制方式) RThreshold 和 SThreshold 属性表示在 OnComm 事件发生之前接收缓沖区或发送缓沖区中可以接收的字符数 以下是通过设置控件属性对串口进行初始化的实例 BOOL CSampleDlg:: PortOpen() { BOOL m_Opened; m_SerialSetCommPort(); // 指定串口号 m_SerialSetSettings(N); // 通信参数设置 m_SerialSetInBufferSize(); // 指定接收缓沖区大小 m_SerialSetInBufferCount(); // 清空接收缓沖区 m_SerialInputMode(); // 设置数据获取方式 m_SerialSetInputLen(); // 设置读取方式 m_Opened=m_SerailSetPortOpen(); // 打开指定的串口 return m_Opened; } 打开所需串口后需要考虑串口通信的时机在接收或发送数据过程中可能需要监视并响应一些事件和错误所以事件驱动是处理串行端口交互作用的一种非常有效的方法使用 OnComm 事件和 CommEvent 属性捕捉并检查通讯事件和错误的值发生通讯事件或错误时将触发 OnComm 事件CommEvent 属性的值将被改变应用程序检查 CommEvent 属性值并作出相应的反应在程序中用ClassWizard为CMSComm控件添加OnComm消息处理函数 void CSampleDlg::OnComm() { switch(m_SerialGetCommEvent()) { case : // 串行口数据接收处理 } } 方法二在单线程中实现自定义的串口通信类 控件简单易用但由于必须拿到对话框中使用在一些需要在线程中实现通信的应用场合控件的使用显得捉襟见肘此时若能够按不同需要定制灵活的串口通信类将弥补控件的不足以下将介绍如何在单线程中建立自定义的通信类 该通信类CSimpleComm需手动加入头文件与源文件其基类为CObject大致建立步骤如下 () 打开串口获取串口资源句柄 通信程序从CreateFile处指定串口设备及相关的操作属性再返回一个句柄该句柄将被用于后续的通信操作并贯穿整个通信过程CreateFile()函数中有几个值得注意的参数设置串口共享方式应设为串口为不可共享设备创建方式必须为OPEN_EXISTING即打开已有的串口对于dwFlagAndAttribute参数对串口有意义的值是FILE_FLAG_OVERLAPPED该标志表明串口采用异步通信模式可进行重叠操作若值为NULL则为同步通信方式在同步方式下应用程序将始终控制程序流直到程序结束若遭遇通信故障等因素将导致应用程序的永久等待所以一般多采用异步通信 ()串口设置 串口打开后其属性被设置为默认值根据具体需要通过调用GetCommState(hComm&dcb)读取当前串口设备控制块DCB(Device Control Block)设置修改后通过SetCommState(hComm&dcb)将其写入再需注意异步读写的超时控制设置 通过COMMTIMEOUTS结构设置超时调用SetCommTimeouts(hComm&timeouts)将结果写入以下是温度监控程序中串口初始化成员函数 BOOL CSimpleComm::Open( ) { DCB dcb; m_hIDComDev=CreateFile( COM GENERIC_READ | GENERIC_WRITENULLOPEN_EXISTINGFILE_ATTRIBUTE_NORMAL|FILE_FLAG_OVE RLAPPED NULL ); // 打开串口异步操作 if( m_hIDComDev == NULL ) return( FALSE ); dcbDCBlength = sizeof( DCB ); GetCommState( m_hIDComDev &dcb ); // 获得端口默认设置 dcbBaudRate=CBR_; dcbByteSize=;dcbParity= NOPARITY; dcbStopBits=(BYTE) ONESTOPBIT; } ()串口读写操作 主要运用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_OverlappedReadhEvent); return ((int)dwBytesRead); } return(); } return ((int)dwBytesRead); 定义全局变量m_Serial作为新建通信类CSimpleComm的对象通过调用类的成员函数即可实现所需串行通信功能与方法一相比方法二赋予串行通信程序设计较大的灵活性端口的读写可选择较简单的查询式或通过设置与外设数据发送时间间隔TimeCycle相同的定时器SetTimer(TimeCycleNULL)进行定时读取或发送 CSampleView:: OnTimer(UINT nIDEvent) { char InputData[]; m_SerialReadData(InputData); // 数据处理 } 若对端口数据的响应时间要求较严格可采用事件驱动I/O读写Windows定义了种串口通信事件较常用的有 EV_RXCHAR: 接收到一个字节并放入输入缓沖区 EV_TXEMPTY: 输出缓沖区中的最后一个字符发送出去 EV_RXFLAG: 接收到事件字符(DCB结构中EvtChar成员)放入输入缓沖区 在用SetCommMask()指定了有用的事件后应用程序可调用WaitCommEvent()来等待事件的发生SetCommMask(hComm)可使WaitCommEvent()中止 方法三 多线程下实现串行通信 方法一二适用于单线程通信在很多工业控制系统中常通过扩展串口连接多个外设各外设发送数据的重复频率不同要求后台实时无差错捕捉采集处理记录各端口数据这就需要在自定义的串行通信类中创建端口监视线程以便在指定的事件发生时向相关的窗口发送通知消息 线程的基本概念可详见VC++参考书目Windows内部的抢先调度程序在活动的线程之间分配CPU时间Win 区分两种不同类型的线程一种是用户界面线程UI(User Interface Thread)它包含消息循环或消息泵用于处理接收到的消息另一种是工作线程(Work Thread)它没有消息循环用于执行后台任务用于监视串口事件的线程即为工作线程 多线程通信类的编写在端口的配置连接部分与单线程通信类相同在端口配置完毕后最重要的是根据实际情况建立多线程之间的同步对象如信号灯临界区事件等相关细节可参考VC++ 中的同步类 一切就绪后即可启动工作线程 CWinThrea *CommThread = AfxBeginThread(CommWatchThread // 线程函数名 (LPVOID) m_pTTYInfo // 传递的参数 THREAD_PRIORITY_ABOVE_NORMAL // 设置线程优先级 (UINT) // 最大堆栈大小 (DWORD) CREATE_SUSPENDED // 创建标志 (LPSECURITY_ATTRIBUTES) NULL); // 安全性标志 同时在串口事件监视线程中 if(WaitCommEvent(pTTYInfo>idComDev&dwEvtMaskNULL)) { if((dwEvtMask & pTTYInfo>dwEvtMask )== pTTYInfo>dwEvtMask) { WaitForSingleObject(pTTYInfo>hPostEventxFFFFFFFF); ResetEvent(pTTYInfo>hPostEvent); // 置同步事件对象为非信号态 ::PostMessage(CSampleViewID_COM_DATA); // 发送通知消息 } } 用PostMessage()向指定窗口的消息队列发送通知消息相应地需要在该窗口建立消息与成员函数间的映射用ON_MESSAGE将消息与成员函数名关联 BEGIN_MESSAGE_MAP(CSampleView CView) //{ { AFX_MSG_MAP(CSampleView)ON_MESSAGE(ID_COM_DATA OnProcessComData) ON_MESSAGE(ID_COM_DATA OnProcessComData)
//} } AFX_MSG_MAPEND_MESSAGE_MAP() 然后在各成员函数中完成对各串口数据的接收处理但必须保证在下一次监测到有数据到来之前能够完成所有的中间处理工作否则将造成数据的捕捉错误 多线程的实现可以使得各端口独立准确地实现串行通信使串口通信具有更广泛的灵活性与严格性且充分利用了CPU时间但在具体的实时监控系统中如何协调多个线程线程之间以何种方式实现同步也是在多线程串行通信程序实现的难点 以VC++ 为工具实现串行通信的三种方法各有利弊 根据不同需要选择合适的方法将达到事半功倍的效果在温度监控系统中笔者采用了方法二在Window Windows 上运行稳定取得了良好的效果 |