1 首先建立MFC项目
2 在项目中插入MSComm控件
(1)选择Project菜单下Add To Project子菜单中的 Components and Control Gallery选项
(2)双击Registered ActiveX Controls项
(3)选择Microsoft Communications Control, version 6.0,,单击Insert按钮将它插入到我们的Project中来,接受缺省的选项。
(4)这时在ClassView视窗中就可以看到CMSComm类了,(注意:此类在ClassWizard中看不到,重构 clw文件也一样),并且在控件工具栏Controls中出现了电话图标(如图1所示),现在要做的是用鼠标将此图标拖到对话框中,程序运行后,这个图标 是看不到的。
3 开始做通信程序
(1)建立对话框
(2)添加控制变量
(3)添加对应函数
1 OnOnCommMscomm1()--------------//从串口接收数据并显示在接受编辑框中
2 在OnInitDialog中添加以下代码
3 OnButtonSend--------------//发送数据
COleSafeArray:
COleSafeArray类是用于处理任意类型和维数的数组的类。COleSafeArray是从OLE VARIANT结构派生而来的。OLE SAFEARRAY成员函数在可以通过COleSafeArray来访问,就象是特别为一维的字节数组所设计的一个成员函数集。
#include 请参阅:COleVariant, CRecordset, CDatabase COleSafeArray类成员
n 构造 COleSafeArray 构造一个COleSafeArray对象
n 操作 Attach 给COleSafeArray对象以存在的VARIANT数组的控制
n Clear 释放基VARIANT中的所有数据
n Detach 将VARIANT数组从COleSafeArray对象中分离出来(这将使数据不会被释放) Win32 API 包装 AccessData 获取一个指向数组数据的指针
n AllocData 为数组分配内存
n AllocDescriptor 为安全数组描述符分配内存
n Copy 创建一个已存在的数组的拷贝
n Create 创建一个安全数组
n Destroy 销毁一个已经存在的数组
n DestroyData 销毁一个安全数组中的数据
n DestroyDescriptor 销毁一个安全数组的描述符
n GetDim 返回数组的维数
n GetElement 获取安全数组中的一个单一元素
n GetElemSize 返回安全数组中一个元素的按字节表示的大小
n GetLBound 返回一个安全数组任一维的下界
n GetUBound 返回一个安全数组任一维的上界
n Lock 增加一个数组的加锁计数,并将一个指向数组数据的指针放到数组描述符中 PtrOfIndex 返回一个指向被索引的元素的指针
n PutElement 将一个单一的元素放入数组中
u Redim 改变一个安全数组的最不重要(最右边)的边界
n UnaccessData 减小一个数组的加锁计数,并使由AccessData获得的指针无效
n Unlock 减小一个数组的加锁以使它能被释放或改变大小
n 一维数组操作 CreateOneDim 创建一个一维的COleSafeArray对象
n GetOneDimSize 返回一个一维的COleSafeArray对象中的元素个数
n ResizeOneDim 改变一个一维的COleSafeArray对象中的元素个数
n 操作 operator = 将一些值(包括SAFEARRAY,VARIANT,COleVariant,或COleSafeArray对象)拷贝到COleSafeArray对象中
n operator == 比较两个不同的数组(SAFEARRAY,VARIANT,ColeVariant,或COleSafeArray对象)
Ø COleSafeArray::COleSafeArray
Ø COleSafeArray();
Ø COleSafeArray( const SAFEARRAY& saSrc, VARTYPE vtSrc ); COleSafeArray( LPCSAFEARRAY psaSrc, VARTYPE vtSrc );
Ø COleSafeArray( const COleSafeArray& saSrc );
Ø COleSafeArray( const VARIANT& varSrc );
Ø COleSafeArray( LPCVARIANT pSrc );
Ø COleSafeArray( const COleVariant& varSrc );
参数:
1) saSrc 要被拷贝到新的COleSafeArray对象中去的已经存在的COleSafeArray对象或SAFEARRAY。
2) vtSrc 新的COleSafeArray对象的VARTYPE。
3) psaSrc 一个指向要被拷贝到新的COleSafeArray对象中去的SAFEARRAY的指针。 varSrc 要被拷贝到新的COleSafeArray对象中去的已经存在的VARIANT或者COleVariant。
4) pSrc 一个指向要被拷贝到新的COleSafeArray对象中去的VARIANT对象的指针。
说明: 所有这些构造函数都创建一个新的COleSafeArray对象。如果没有参数,则创建的是一个空的COleSafeArray对象(VT_EMPTY)。如果COleSafeArray是从另一个数组拷贝来的,并且这个数组的VARTYPE并不是完全确定的(一个COleSafeArray,COleVariant,或者VARIANT),则源数组中的VARTYPE被保留,并且不需要说明。如果COleSafeArray是从另一个数组拷贝而来,并且该数组的VARTYPE是不知道的,则VARTYPE必须用vtSrc参数来指定。 如果出错,则函数抛出一个CMemoryException或COleException
在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 不支持设备的波特率
11楼
8008 指定的字节大小无效
8009 缺省参数错误
8010 硬件不可用(被其他设备锁住)
8011 函数不能分配队列
ComNoOpen 8012 设备没有打开
8013 设备已经打开
8014 不能使用通信通知
ComSetCommStateFailed 8015 不能设置通信状态
8016 不能设置通信事件屏蔽
ComPortNotOpen 8018 该存在只在端口打开是有效
8019 设备忙
ComReadError 8020 通信设备读错误
ComDCBError 8021 检取端口设备控制块时出现内部错误
注意在使用的时候一定要保证两个通讯串口的设置是相同的,否则受到的信息将会产生错误!由于取值位数的不同,有可能发送的信息要读很多次才能组合成需要的信息!
(1)MSComm控件的调出
VB6.0的MSComm控件并不会主动出现在工具箱中,当需要它时,让它出现在工具箱中的步骤如下:
①选择菜单上的“工程”;
②在“工程”菜单中选择“部件”;
③出现对话框后,在可勾选的项目中勾选Microsoft Comm Contro l6.0;
④按下“确定”按钮,即可在工具箱中见到“电话盒”的图标,就可以将此控件加载进行串行通信。
(2)MSComm属性
MSComm属性很多,其重要的属性说明如下:
(1)CommPort:设定或返回通信端口号。端口号由1开始往上递增,最大值是16。
(2)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。
(3)PortOpen:设定并返回通信端口的状态。
使用串行端口之前必须先打开该端口(PortOpen=True),而在使用完毕后必须关闭该端口(PortOpen=False)。
(4)Input:从输入缓冲区返回并清除字符。这是一种FIFO(FirstinFirstOut)机制。如Buffer$=MSComm1.Input,表示将输入缓冲区的字符读入Buffer字符串变量中。
(5)Output:将一个字符或字符串写入传输缓冲区,如MSComm1.Output=“ABCD”。此即将ABCD4个字符通过串行端口传送出去。
(6)InBufferCount:传回在接收缓冲区中的字符数,是指已接收,并在接收缓冲区等待读取的字符数。
(7)OutBufferCount: 设置或返回输入缓冲区内等待读取的字节个数。
Void CMSComm::SetOutBufferCount(short nNewValue);
Short CMSComm::GetOutBufferCount();
说明:当设置InBufferCount属性值为0时,可以清空发送缓冲区
(8)InputMode:设定和返回类型。该属性设为0时,数据通过Input属性以文本方式取回,如设为1,则数据通过Input属性以二进制方式取回。
(9)InputLen:设置并返回Input属性从接收缓冲区读取的字符数。
Void CMSComm::SetInputLen(short nNewValue);
Short CMSComm::GetInputLen();
说明:InputLen属性的缺省值是0.设置InputLen为0时,使用Input将使MSComm控件读取接收缓冲区中全部的内容。
若接受缓冲区中InputLen字符无效,Input属性返回一个零长度字符串(“”)。在使用Input前,用户可以选择检查InBufferCount来确定缓冲区中是否已有需要数目的字符。该属性在从输出格式为定长数据的机器读取数据时非常有用。
(10)InBufferSize:设置或返回输入缓冲区的大小。
VoidCMSComm::SetInBufferSize(short nNewValue);
Short CMSComm::GetInBufferSize();
说明:设置值的缺省值(默认值大小为1024字节(byte))。
(11)OutBufferSize: 设置或返回发送缓冲区内的大小。
VoidCMSComm::SetOutBufferSize(short nNewValue);
Short CMSComm::GetOutBufferSize();
说明:设置值的缺省值,默认值大小为512字节(byte),此值太小,缓冲区数据易溢出,太大会占不必要的内存。
3.MSComm控件进行串口编程步骤:
1. 在建立的程序工程中插入MSComm控件。
2. 添加MSComm控件ID 的控制变量(或者对象)。
3. 对串口进行初始化,设置MSComm控件的属性。
4. 添加串口事件的消息处理函数OnComm()函数,在函数中根据应用需要,编写数据处理代码
5. 编写串口发送等其他代码。
6. 关闭串口
串口通信原理
看串口通信原理,(也可以说大多数通信原理也是如此)。通信首先要有个通信,可以简单的把通信看成一个小桶,发送方住水桶里装水,接收方从水桶中取水。如果你要和对方通信首先需要将桶盖打开,再将水装入到桶中,这时接收方才能够从桶中取到水。这里就存在着一定的问题,
1, 如果桶盖还没有打开,发送方已经发送了。这时接收方再从桶中取水,肯定取的水不对,会使一部分缺失了。解决方式就是让桶盖打开再往其中加水。
2, 但是桶盖何时打开,发送方何时发送,这个不好把握。
解决方法:接收方接到数据时,要返回一个应答标志,告诉发送方我已经取到数据了,而且是取得到正确数据才应答,否则不理会,继续取数据。或者一直查询,直到与发送方发来的数据一致再停止取数据。
一般的,进行串口通信总有一个是主动方一个是被动方,而且二者传输数据时,会有一定的协商好的数据格式,二者发送接收都按照此数据格式进行。