一个印度人写的VC串口类CSerialCom(有串口基础介绍)

2024-02-17 04:32

本文主要是介绍一个印度人写的VC串口类CSerialCom(有串口基础介绍),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

http://www.vc-rs232.com/html/VC_SSCOM_Control/2011/0117/34.html

 

一个印度人写的VC串口类CSerialCom(有串口基础介绍)

软件介绍

  一个由印度人编写的VC串口类(也是一种VC串口控件),他还配合这个类写了VC 串口通信方面的一些基础知识,如怎么用VC打开串口,如何对串口进行配置,读串口、写串口等。

  这个类有点特别,它没有使用事件驱动原理,它是以查询方式工作的。

简介:  

  对没有接触过串口通信的VC程序员来说显得非常困难,很久以前我在 codeguru.com 上搜索过串口通信相关信息得到了非常大的帮助,从那时起能编写一个简单易用的VC 串口类是我的梦想。

  经过七个月在串口通信编程方面实践经验后,我编写了一个基于API实现的简单串口类,在介绍此串口类之前先介绍一下VC 串口通信方面的基础知识。

串口通信基础:

  串口通信每个字节的传输以串行的方式进行,传输时低位先被发送,一个“包”由“开始位”+“数据位”+“奇偶校验位”(不是必需)+“停止位” 组成。

  奇偶校验位是可选的,它用来进行错误检测,您可以在软件里设置是否启用奇偶校验,而且还可以选择启用哪种校验方式,如“奇”校验(ODD)或“偶”校验(EVEN)。

  PC机通过串口发送和接收数据的流程如下:

 1、

打开串口

 2、配置串口通信参数,如:波特率、校验方式、数据位数等
 3、设置通信超时时间
 4、写数据
 5、读数据
 6、关闭串口

用VC打开串口:

  打开串口可以用API函数 CreateFile() 来实现,打开串口有两种方式,分别为重叠I/O(OVERLAPPED)和非重叠(NON-OVERLAPPED)方式(其实这两种方式分别对应串口的异步和同步通信方式-VC串口通信技术网注)。CSerialCom 类工作于非重叠(NON-OVERLAPPED)模式(即同步通信模式),有关更多 OVERLAPPED NON-OVERLAPPED 方面的消息可查询MSDN。

串口配置:

  VC 串口通信程序编写最重要的课题就是如何利用 DCB 结构来配置串口,对 DCB 结构填充不正确是大多数人常反犯的毛病,通常串口通信程序编写好后出现这样那样的问题就是因为这个结构没有正确填充所致。在利用 CreateFile() 函数打开串口时就需要我们对串口的波特率、校验方式、数据位、停止位等进行配置。

设置超时时间:

  每次打开串口时都必须利用 COMMTIMEOUTS 结构设置超时时间,如果未设置此结构,通信将以默认或上次打开此串口时设置的超时时间为准。

写串口:

  WriteFile() 函数可实现这一功能,在执行这一动作之前必须先打开并配置好串口。

读串口:

  利用 ReadFile() 可实现这一功能,同样,在执行这一动作之前必须先打开并配置好串口。

关闭串口:

  不再使用已经打开的串口后必须关闭它以将此串口资源释放出来,这样其它程序才可能再使用此串口资源,外于非重叠I/O(NON-OVERLAPPED)模式下的串口在打开期间是不能被其它程序或同一程序内的其它线程访问的。使用 CloseHandle() 函数可以关闭串口, CloseHandle() 函数只有一个参数,它是由 CreateFile() 打开串口时返回的设置句柄。

CSerialCom 串口类

  CSerialCom 类使用六个函数来实现上面提到的功能,它们分别是:

  1. BOOL CSerialCom::OpenPort(CString portname) 
  2. portname= "//./" + portname; 
  3. hComm = CreateFile(portname, 
  4.                       GENERIC_READ | GENERIC_WRITE, 
  5.                       0, 
  6.                       0, 
  7.                       OPEN_EXISTING, 
  8.                       0, 
  9.                       0); 
  10. if(hComm==INVALID_HANDLE_VALUE){ 
  11.     return false;} 
  12.     else
  13.     return true

  OpenPort()成员函数用来打开串口,它需要的参数只有一个,就是串口名,如“COM1”,“COM2” 。

  1. BOOL CSerialCom::ConfigurePort(DWORD BaudRate, BYTE ByteSize,  
  2.                                DWORD fParity, BYTE Parity, BYTE StopBits) 
  3.     if((m_bPortReady = GetCommState(hComm, &m_dcb))==0) 
  4.     { 
  5.         MessageBox("GetCommState Error","Error",MB_OK+MB_ICONERROR); 
  6.         CloseHandle(hComm); 
  7.         return false
  8.     } 
  9.      
  10.     m_dcb.BaudRate =BaudRate; 
  11.     m_dcb.ByteSize = ByteSize; 
  12.     m_dcb.Parity =Parity ; 
  13.     m_dcb.StopBits =StopBits; 
  14.     m_dcb.fBinary=TRUE; 
  15.     m_dcb.fDsrSensitivity=false
  16.     m_dcb.fParity=fParity; 
  17.     m_dcb.fOutX=false
  18.     m_dcb.fInX=false
  19.     m_dcb.fNull=false
  20.     m_dcb.fAbortOnError=TRUE; 
  21.     m_dcb.fOutxCtsFlow=FALSE; 
  22.     m_dcb.fOutxDsrFlow=false
  23.     m_dcb.fDtrControl=DTR_CONTROL_DISABLE; 
  24.     m_dcb.fDsrSensitivity=false
  25.     m_dcb.fRtsControl=RTS_CONTROL_DISABLE; 
  26.     m_dcb.fOutxCtsFlow=false
  27.     m_dcb.fOutxCtsFlow=false
  28.     m_bPortReady = SetCommState(hComm, &m_dcb); 
  29.     if(m_bPortReady ==0) 
  30.     { 
  31.         MessageBox("SetCommState Error","Error",MB_OK+MB_ICONERROR); 
  32.         CloseHandle(hComm); 
  33.         return false
  34.     } 
  35.     return true

  ConfigurePort() 函数对串口地配置,它所需参数如下所示:

  DWORD BaudRate

  DWORD BaudRate 用来描述串口通信使用的波特率,如此参数为 CBR_9600 时,波特率为9600bps,PC机支持的标准波特率有:CBR_110 ,CBR_300 ,CBR_600 ,CBR_1200,CBR_2400,CBR_4800,CBR_9600,CBR_14400, CBR_19200,CBR_38400,CBR_56000,CBR_57600,CBR_115200,CBR_128000,CBR_256000

  BYTE ByteSize

  此参数描述数据位位数,标准值为8或4。

  DWORD fParity

  奇偶校验开关,如果此参数为真(TRUE)则开启奇偶校验,为假(FALSE)则关闭奇偶校验。

  BYTE Parity

  校验方式,可选方式如下:

 

EVENPARITY

 MARKPARITY
 NOPARITY
 ODDPARITY
 SPACEPARITY

  BYTE StopBits

  停止位位数,可取值如下:

 

ONESTOPBIT

 ONE5STOPBITS
 TWOSTOPBITS

备注:ConfigurePort() 函数是假定串口的流控制是由硬件来完成的,软件在收发数据过程中不检测CTS/RTS 和 Xon/Xoff 状态,您可以修改DCB结构来启用软件流控制。

  1. BOOL CSerialCom::SetCommunicationTimeouts(DWORD ReadIntervalTimeout,  
  2.                                           DWORD ReadTotalTimeoutMultiplier,  
  3.                                           DWORD ReadTotalTimeoutConstant,  
  4.                                           DWORD WriteTotalTimeoutMultiplier,  
  5.                                           DWORD WriteTotalTimeoutConstant) 
  6.     if((m_bPortReady = GetCommTimeouts (hComm, &m_CommTimeouts))==0) 
  7.         return false
  8.     m_CommTimeouts.ReadIntervalTimeout =ReadIntervalTimeout; 
  9.     m_CommTimeouts.ReadTotalTimeoutConstant =ReadTotalTimeoutConstant; 
  10.     m_CommTimeouts.ReadTotalTimeoutMultiplier =ReadTotalTimeoutMultiplier; 
  11.     m_CommTimeouts.WriteTotalTimeoutConstant = WriteTotalTimeoutConstant; 
  12.     m_CommTimeouts.WriteTotalTimeoutMultiplier =WriteTotalTimeoutMultiplier; 
  13.     m_bPortReady = SetCommTimeouts (hComm, &m_CommTimeouts); 
  14.      
  15.     if(m_bPortReady ==0) 
  16.     { 
  17.         MessageBox("StCommTimeouts function failed"
  18.                    "Com Port Error",MB_OK+MB_ICONERROR); 
  19.         CloseHandle(hComm); 
  20.         return false
  21.     } 
  22.     return true
  23. }

  SetCommunicationTimeouts() 成员函数用来设置读/写超时时间,它所需参数如下:

  DWORD ReadIntervalTimeout

  设置读串口时,收到两个字符的最大时间间隔,单位为毫秒。在执行ReadFile() 期间,一但收到两个字符的时间间隔超过这一设定值,ReadFile()立即返回。如果这一参数设置为零,则表明这一特性未启用。

  ReadTotalTimeoutConstant

  常量,用来计算读操作总超时时间。每个读操作过程中,这个参数被加到ReadTotalTimeoutMultiplier。如果ReadTotalTimeoutMultiplierReadTotalTimeoutConstant两个参数都为零则表明总超时计算功能没有启用。

  ReadTotalTimeoutMultiplier

  用来保存读操作总超时时间的

  WriteTotalTimeoutConstant

  常量,用来计算写操作总超时时间。与ReadTotalTimeoutConstant差不多。

  WriteTotalTimeoutMultiplier

  用来保存写操作总超时时间的,如果WriteTotalTimeoutMultiplierWriteTotalTimeoutConstant两参数均为零,表明写操作总超时时间特性不被启用。

  例如,您需要传送一块数据包,设定了写操作超时时间为500ms(每个字符间发送时间最大间隔),您可以用设置超时成员函数来设定,内容为:SetCommunicationTimeouts(0,500,0,0,0);如果设置成功,函数返回真,否则返回假。

  1. BOOL CSerialCom::WriteByte(BYTE bybyte) 
  2.     iBytesWritten=0; 
  3.     if(WriteFile(hComm,&bybyte,1,&iBytesWritten,NULL)==0) 
  4.         return false
  5.     else  
  6.         return true
  7. }

  WriteByte() 成员函数用来向串口写数据。

  1. BOOL CSerialCom::ReadByte(BYTE &resp) 
  2.     BYTE rx; 
  3.     resp=0; 
  4.     DWORD dwBytesTransferred=0; 
  5.     if (ReadFile (hComm, &rx, 1, &dwBytesTransferred, 0)) 
  6.     { 
  7.         if (dwBytesTransferred == 1) 
  8.         { 
  9.             resp=rx; 
  10.             return true
  11.          } 
  12.     } 
  13.     return false
  14. }

  

ReadByte() 成员函数实现读串口功能,如果您知道有数据发送来了,可能使用 ReadByte() 来读取数据,如果不知道何时有数据发送过来,可以周期性调用此函数,当没收到数据超过超时时间后, ReadByte() 自动返回。在实际应用中,通信双方可能会以某种协议进行的,此时一般有表明通信数据包结束的标志,如3964协议的结束符是‘ETX’,您可能通过它来判断是否传输结束。

  1. void CSerialCom::ClosePort() 
  2. CloseHandle(hComm); 
  3. return

 

  ClosePort() 成员函数用来关闭一个已经打开的串口。

如何使用CSerialCom类

  使用CSerialCom类前的准备工作:

 1、

复制SerialCom.h 和 SerialCom.cpp两个文件到工程所在目录

 2、在VC工程中导入这两个文件
 3、加入#include "SerialCom.h"句代码到程序适当位置
 4、创建CSerialCom类的一个实例

  现在就可以调用CSerialCom类成员函数来进行串口通信了,操作流程如下:

  1. // 打开串口。 需要检查函数返回值来判断串口是否正确打开
  2. port.OpenPort( );  
  3. // 配置串口。需要检查函数返回值来判断否正确配置
  4. port.ConfigurePort( );   
  5. // 设置超时时间。需要检查函数返回值来判断否正确配置
  6. port.SetCommunicationTimeouts( );  
  7.       
  8. // 写串口,由于一次只能写一个字节,一般需要多次调用此函数才能将需要发送的字符串发送完
  9. port.WriteByte();  
  10.      
  11. // 读串口,由于一次只能读一个字节,一般需要多次调用此函数才能将需要读取的字符串读完
  12. port.ReadByte( );  
  13. // 关闭串口
  14. port.ClosePort();  

 

这篇关于一个印度人写的VC串口类CSerialCom(有串口基础介绍)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/716683

相关文章

性能测试介绍

性能测试是一种测试方法,旨在评估系统、应用程序或组件在现实场景中的性能表现和可靠性。它通常用于衡量系统在不同负载条件下的响应时间、吞吐量、资源利用率、稳定性和可扩展性等关键指标。 为什么要进行性能测试 通过性能测试,可以确定系统是否能够满足预期的性能要求,找出性能瓶颈和潜在的问题,并进行优化和调整。 发现性能瓶颈:性能测试可以帮助发现系统的性能瓶颈,即系统在高负载或高并发情况下可能出现的问题

水位雨量在线监测系统概述及应用介绍

在当今社会,随着科技的飞速发展,各种智能监测系统已成为保障公共安全、促进资源管理和环境保护的重要工具。其中,水位雨量在线监测系统作为自然灾害预警、水资源管理及水利工程运行的关键技术,其重要性不言而喻。 一、水位雨量在线监测系统的基本原理 水位雨量在线监测系统主要由数据采集单元、数据传输网络、数据处理中心及用户终端四大部分构成,形成了一个完整的闭环系统。 数据采集单元:这是系统的“眼睛”,

Hadoop数据压缩使用介绍

一、压缩原则 (1)运算密集型的Job,少用压缩 (2)IO密集型的Job,多用压缩 二、压缩算法比较 三、压缩位置选择 四、压缩参数配置 1)为了支持多种压缩/解压缩算法,Hadoop引入了编码/解码器 2)要在Hadoop中启用压缩,可以配置如下参数

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

图神经网络模型介绍(1)

我们将图神经网络分为基于谱域的模型和基于空域的模型,并按照发展顺序详解每个类别中的重要模型。 1.1基于谱域的图神经网络         谱域上的图卷积在图学习迈向深度学习的发展历程中起到了关键的作用。本节主要介绍三个具有代表性的谱域图神经网络:谱图卷积网络、切比雪夫网络和图卷积网络。 (1)谱图卷积网络 卷积定理:函数卷积的傅里叶变换是函数傅里叶变换的乘积,即F{f*g}

【Linux 从基础到进阶】Ansible自动化运维工具使用

Ansible自动化运维工具使用 Ansible 是一款开源的自动化运维工具,采用无代理架构(agentless),基于 SSH 连接进行管理,具有简单易用、灵活强大、可扩展性高等特点。它广泛用于服务器管理、应用部署、配置管理等任务。本文将介绍 Ansible 的安装、基本使用方法及一些实际运维场景中的应用,旨在帮助运维人员快速上手并熟练运用 Ansible。 1. Ansible的核心概念

C++——stack、queue的实现及deque的介绍

目录 1.stack与queue的实现 1.1stack的实现  1.2 queue的实现 2.重温vector、list、stack、queue的介绍 2.1 STL标准库中stack和queue的底层结构  3.deque的简单介绍 3.1为什么选择deque作为stack和queue的底层默认容器  3.2 STL中对stack与queue的模拟实现 ①stack模拟实现

AI基础 L9 Local Search II 局部搜索

Local Beam search 对于当前的所有k个状态,生成它们的所有可能后继状态。 检查生成的后继状态中是否有任何状态是解决方案。 如果所有后继状态都不是解决方案,则从所有后继状态中选择k个最佳状态。 当达到预设的迭代次数或满足某个终止条件时,算法停止。 — Choose k successors randomly, biased towards good ones — Close

VC网络协议

// PCControlDlg.cpp : 实现文件//#include "stdafx.h"#include "PCControl.h"#include "PCControlDlg.h"#include "afxdialogex.h"#ifdef _DEBUG#define new DEBUG_NEW#endif// 用于应用程序“关于”菜单项的 CAboutDlg 对话框#ifde

Mysql BLOB类型介绍

BLOB类型的字段用于存储二进制数据 在MySQL中,BLOB类型,包括:TinyBlob、Blob、MediumBlob、LongBlob,这几个类型之间的唯一区别是在存储的大小不同。 TinyBlob 最大 255 Blob 最大 65K MediumBlob 最大 16M LongBlob 最大 4G