vc串口通讯控件MSComm编程详解.doc_第1页
vc串口通讯控件MSComm编程详解.doc_第2页
vc串口通讯控件MSComm编程详解.doc_第3页
vc串口通讯控件MSComm编程详解.doc_第4页
vc串口通讯控件MSComm编程详解.doc_第5页
已阅读5页,还剩22页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

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 奇偶校验错。硬件检测到奇偶校验错误7ComEventTxFull 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); /选择COM1m_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_SendDatai); 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(k0) m_input=m_MSComm.GetInput(); str=(char*)(unsigned char*)m_input.parray-pvData; i=0; str1=str; while(i ) i+; str1+; *str1=post.content; str2=(const char*)str; /清除字符串中的不必要字符 m_RcvData=(const char *)str; /数据显示处理m_disp+=m_RcvData;UpdateData(false);目 次控件两种处理通讯的方式MSComm控件两种处理通讯的方式CommPort属性RThreshold 属性CTSHolding 属性SThreshold 属性CDHolding 属性DSRHolding 属性Settings 属性InputLen 属性EOFEnable 属性Handshake 常数OnComm 常数InputMode 常数错误消息 MSComm 控件通过串行端口传输和接收数据,为应用程序提供串行通讯功能。MSComm控件在串口编程时非常方便,程序员不必去花时间去了解较为复杂的API函数,而且在VC、VB、Delphi等语言中均可使用。Microsoft Communications Control(以下简称MSComm)是Microsoft公司提供的简化Windows下串行通信编程的ActiveX控件,它为应用程序提供了通过串行接口收发数据的简便方法。具体的来说,它提供了两种处理通信问题的方法:一是事件驱动(Eventdriven)方法,一是查询法。控件两种处理通讯的方式1.MSComm控件两种处理通讯的方式MSComm控件提供下列两种处理通讯的方式:事件驱动方式和查询方式。1.1 事件驱动方式事件驱动通讯是处理串行端口交互作用的一种非常有效的方法。在许多情况下,在事件发生时需要得到通知,例如,在串口接收缓冲区中有字符,或者 Carrier Detect (CD) 或 Request To Send (RTS) 线上一个字符到达或一个变化发生时。在这些情况下,可以利用 MSComm 控件的 OnComm 事件捕获并处理这些通讯事件。OnComm 事件还可以检查和处理通讯错误。所有通讯事件和通讯错误的列表,参阅 CommEvent 属性。在编程过程中,就可以在OnComm事件处理函数中加入自己的处理代码。这种方法的优点是程序响应及时,可靠性高。每个MSComm 控件对应着一个串行端口。如果应用程序需要访问多个串行端口,必须使用多个 MSComm 控件。1.2 查询方式查询方式实质上还是事件驱动,但在有些情况下,这种方式显得更为便捷。在程序的每个关键功能之后,可以通过检查 CommEvent 属性的值来查询事件和错误。如果应用程序较小,并且是自保持的,这种方法可能是更可取的。例如,如果写一个简单的电话拨号程序,则没有必要对每接收一个字符都产生事件,因为唯一等待接收的字符是调制解调器的“确定”响应。2.MSComm 控件的常用属性MSComm 控件有很多重要的属性,但首先必须熟悉几个属性。CommPort 设置并返回通讯端口号。Settings 以字符串的形式设置并返回波特率、奇偶校验、数据位、停止位。PortOpen 设置并返回通讯端口的状态。也可以打开和关闭端口。Input 从接收缓冲区返回和删除字符。Output 向传输缓冲区写一个字符串。下面分别描述:CommPort属性 设置并返回通讯端口号。语法 object.CommPortvalue (value 一整型值,说明端口号。)说明 在设计时,value 可以设置成从 1 到 16 的任何数(缺省值为 1)。但是如果用 PortOpen 属性打开一个并不存在的端口时,MSComm 控件会产生错误 68(设备无效)。注意:必须在打开端口之前设置 CommPort 属性。RThreshold 属性:在 MSComm 控件设置 CommEvent 属性为 comEvReceive 并产生 OnComm 之前,设置并返回的要接收的字符数。语法 object.Rthreshold = value (value 整型表达式,说明在产生 OnComm 事件之前要接收的字符数。)说明 当接收字符后,若 Rthreshold 属性设置为 0(缺省值)则不产生 OnComm 事件。例如,设置 Rthreshold 为 1,接收缓冲区收到每一个字符都会使 MSComm 控件产生 OnComm 事件。CTSHolding 属性:确定是否可通过查询 Clear To Send (CTS) 线的状态发送数据。Clear To Send 是调制解调器发送到相联计算机的信号,指示传输可以进行。该属性在设计时无效,在运行时为只读。语法: object.CTSHolding(Boolean)Mscomm 控件的 CTSHolding 属性设置值:True Clear To Send 线为高电平。False Clear To Send 线为低电平。说明:如果 Clear To Send 线为低电平 (CTSHolding = False) 并且超时时,MSComm 控件设置 CommEvent 属性为 comEventCTSTO (Clear To Send Timeout) 并产生 OnComm 事件。Clear To Send 线用于 RTS/CTS (Request To Send/Clear To Send) 硬件握手。如果需要确定 Clear To Send 线的状态,CTSHolding 属性给出一种手工查询的方法。详细信息 有关握手协议,请参阅 Handshaking 属性。SThreshold 属性: MSComm 控件设置 CommEvent 属性为 comEvSend 并产生 OnComm 事件之前,设置并返回传输缓冲区中允许的最小字符数。语法 object.SThreshold = value value 整形表达式,代表在 OnComm 事件产生之前在传输缓冲区中的最小字符数。说明:若设置 Sthreshold 属性为 0(缺省值),数据传输事件不会产生 OnComm 事件。若设置 Sthreshold 属性为 1,当传输缓冲区完全空时,MSComm 控件产生 OnComm 事件。如果在传输缓冲区中的字符数小于 value,CommEvent 属性设置为 comEvSend,并产生 OnComm 事件。comEvSend 事件仅当字符数与 Sthreshold 交叉时被激活一次。例如,如果 Sthreshold 等于 5,仅当在输出队列中字符数从 5 降到 4 时,comEvSend 才发生。如果在输出队列中从没有比 Sthreshold 多的字符,comEvSend 事件将绝不会发生。Handshake 常数常数 值 描述comNone 0 无握手。comXonXoff 1 XOn/Xoff 握手。comRTS 2 Request-to-send/clear-to-send 握手。comRTSXOnXOff 3 Request-to-send 和 clear-to-send 握手皆可。OnComm 常数常数 值 描述comEvSend 1 发送事件。comEvReceive 2 接收事件。comEvCTS 3 clear-to-send 线变化。comEvDSR 4 data-set ready 线变化。comEvCD 5 carrier detect 线变化。comEvRing 6 振铃检测。comEvEOF 7 文件结束。Error 常数常数 值 描述comEventBreak 1001 接收到中断信号comEventCTSTO 1002 Clear-to-send 超时comEventDSRTO 1003 Data-set ready 超时comEventFrame 1004 帧错误comEventOverrun 1006 端口超速comEventCDTO 1007 Carrier detect 超时comEventRxOver 1008 接收缓冲区溢出comEventRxParity 1009 Parity 错误comEventTxFull 1010 传输缓冲区满comEventDCB 1011 检索端口 设备控制块 (DCB) 时的意外错误InputMode 常数常数 值 描述comInputModeText 0 (缺省) 通过 Input 属性以文本方式取回数据。comInputModeBinary 1 通过 Input 属性以二进制方式检取回数据。CDHolding 属性:通过查询 Carrier Detect (CD) 线的状态确定当前是否有传输。Carrier Detect 是从调制解调器发送到相联计算机的一个信号,指示调制解调器正在联机。该属性在设计时无效,在运行时为只读。语法 object.CDHolding设置值:CDHolding 属性的设置值为:设置 描述True Carrier Detect 线为高电平False Carrier Detect 线为低电平说明:注意当 Carrier Detect 线为高电平 (CDHolding = True) 且超时时,MSComm 控件设置CommEvent 属性为 comEventCDTO(Carrier Detect 超时错误),并产生 OnComm 事件。注意 在主机应用程序中捕获一个丢失的传输是特别重要的,例如一个公告板,因为呼叫者可以随时挂起(放弃传输)。Carrier Detect 也被称为 Receive Line Signal Detect (RLSD)。数据类型 BooleanDSRHolding 属性:确定 Data Set Ready (DSR) 线的状态。Data Set Ready 信号由调制解调器发送到相连计算机,指示作好操作准备。该属性在设计时无效,在运行时为只读。语法:object.DSRHoldingobject 所在处表示对象表达式,其值是“应用于”列表中的对象。DSRHolding 属性返回以下值:值 描述True Data Set Ready 线高False Data Set Ready 线低说明:当 Data Set Ready 线为高电平 (DSRHolding = True) 且超时时,MSComm 控件设置 CommEvent 属性为 comEventDSRTO(数据准备超时)并产生 OnComm 事件。当为 Data Terminal Equipment (DTE) 机器写 Data Set Ready/Data Terminal Ready 握手例程时该属性是十分有用的。数据类型:BooleanSettings 属性: 设置并返回波特率、奇偶校验、数据位、停止位参数。语法: object.Settings = value说明:当端口打开时,如果 value 非法,则 MSComm 控件产生错误 380(非法属性值)。Value 由四个设置值组成,有如下的格式:BBBB,P,D,SBBBB 为波特率,P 为奇偶校验,D 为数据位数,S 为停止位数。value 的缺省值是:9600,N,8,1InputLen 属性:设置并返回 Input 属性从接收缓冲区读取的字符数。语法 object.InputLen = valueInputLen 属性语法包括下列部分:value 整型表达式,说明 Input 属性从接收缓冲区中读取的字符数。说明:InputLen 属性的缺省值是 0。设置 InputLen 为 0 时,使用 Input 将使 MSComm 控件读取接收缓冲区中全部的内容。若接收缓冲区中 InputLen 字符无效,Input 属性返回一个零长度字符串 ()。在使用 Input 前,用户可以选择检查 InBufferCount 属性来确定缓冲区中是否已有需要数目的字符。该属性在从输出格式为定长数据的机器读取数据时非常有用。EOFEnable 属性:确定在输入过程中 MSComm 控件是否寻找文件结尾 (EOF) 字符。如果找到 EOF 字符,将停止输入并激活 OnComm 事件,此时 CommEvent 属性设置为 comEvEOF,语法:object.EOFEnable = value EOFEnable 属性语法包括下列部分:value 布尔表达式,确定当找到 EOF 字符时,OnComm 事件是否被激活,如“设置值”中所描述。value 的设置值:True 当 EOF 字符找到时 OnComm 事件被激活。False (缺省)当 EOF 字符找到时 OnComm 事件不被激活。说明:当 EOFEnable 属性设置为 False,OnComm 控件将不在输入流中寻找 EOF 字符。错误消息(MS Comm 控件)下表列出 MSComm 控件可以捕获的错误:值 描述380 无效属性值 comInvalidPropertyValue383 属性为只读 comSetNotSupported394 属性为只读 comGetNotSupported8000 端口打开时操作不合法 comPortOpen8001 超时值必须大于 08002 无效端口号 comPortInvalid8003 属性只在运行时有效8004 属性在运行时为只读8005 端口已经打开 comPortAlreadyOpen8006 设备标识符无效或不支持该标识符8007 不支持设备的波特率8008 指定的字节大小无效8009 缺省参数错误8010 硬件不可用(被其它设备锁定)8011 函数不能分配队列8012 设备没有打开 comNoOpen8013 设备已经打开8014 不能使用 comm 通知8015 不能设置 comm 状态 comSetCommStateFailed8016 不能设置 comm 事件屏蔽8018 仅当端口打开时操作才有效 comPortNotOpen8019 设备忙8020 读 comm 设备错误 comReadError8021 为该端口检索设备控制块时的内部错误 comDCBError1 引言近年来,随着微处理器、计算机和数字通信技术的迅猛发展,计算机控制已扩展到了几乎所有的工业领域。它不仅以其良好的性能满足了工业生产的广泛需要,而且将通信技术与信息处理技术融为一体,成为具有逻辑控制功能、过程控制功能、运动控制功能、数据处理功能、联网通信功能的多功能控制器。在PLC组成的控制系统中,一般由PLC作为下位机,完成数据采集、状态判别、输出控制等,上位机(微型计算机、工业控制机)完成采集数据信息的存储、分析处理、人机界面的交互以及打印输出,以实现对系统的实时监控。这种监控系统充分利用了微型机和PLC各自的特点,实现了优势互补。其中的技术关键是实现PLC与计算机的互联通信。2 通信方式和通信协议2.1通信方式目前PLC和PC机的互联通信方式有以下几种:(1) 通过PLC开发商提供的系统协议和网络适配器,构成特定公司产品的内部网络,其通信协议不公开。互联通信必须使用开发商提供的上位机组态软件,并采用支持相应协议的外设。这种方式其显示画面和功能往往难以满足用户的具体需要;(2) 购买目前通用的上位机组态软件。这种方式除了要增加系统投资以外,其运用的灵活性也受到一定限制;(3) 利用PLC厂商所提供的标准通信端口和由用户自定义的自由端口通信方式。这种方式不需要增加投资,具备较好的灵活性,特别适合小规模控制系统。S7-200系列PLC的通信接口是与RS-485兼容的9针D型连接器,符合EN 50170。附表给出了通信口的引脚分配。PC机的标准串口为RS-232,西门子公司提供的PC/PPI电缆带有RS232/RS485电平转换器,因此在不增加任何硬件的情况下,可以很方便地将PLC和PC机互联。2.2通信协议S7-200支持多种通信协议,主要有:(1) 点对点接口协议(PPI)PPI是主/从协议,网络上的S7-200 CPU均为从站,其他CPU、SIMATIC编程器或TD200为主站;(2) 多点接口协议(MPI)MPI是集成在西门子公司的可编程序控制器、操作员界面和编程器上的集成通信接口,用于建立小型的通信网络。最多可接32个节点,典型数据长度为64字节,最大距离100m;(3) Profibus协议Profibus协议用于分布式I/O设备(远程I/O)的高速通信。许多厂家生产类型众多的PROFIBUS设备,如简单的输入/输出模块、电机控制器和可编程序控制器;(4) 用户定义协议(自由端口模式)通过使用接收中断、发送中断、字符中断、发送指令(XMT)和接收指令(RCV),自由端口通信可以控制S7-200 CPU通信口。通过SMB30,允许在CPU处于RUN模式时通信口采用自由端口模式。CPU处于STOP模式时,停止自由端口通信,通信口强制转换为PPI协议模式。自由端口模式为计算机与S7-200 CPU之间的通信提供了一种廉价与灵活的方法。计算机与PLC通信时,为了避免各方争用信道,一般采用主从方式,即计算机为主机,PLC为从机,只有主机才有权主动发送请求报文,从机收到后返回响应报文。3 PLC端通信编程PLC的通信编程首先是对串口初始化。对S7-200的初始化是通过对特殊标志位SMB30写入通信控制字,设置端口0通信的波特率,奇偶校验位、停止位和字符长度。SMB130用于端口1的设置。显然,这些设定必须与PC机设定相一致。S7-200系列有专用的发送指令XMT(Transmit),通过指定的通信端口(PORT),发送存储在数据缓冲区(TBL)中的信息。接收指令RCV(Receive)初始化或终止接收信息的服务,通过指定的通信端口(PORT),接收信息并存储在数据缓冲区(TBL)中。为提高通信可靠性可以采用异或校验(或求和校验)。使用字符中断方式接收数据,以起始字符作为接收报文的开始,部分程序如下:/主程序LD SM0.0 MOVB 16#05, SMB30 /19200bps,8位数据,无奇偶校验,1位停止位ATCH INT_0, 8 /出现接收字符中断时执行INT_0ENI /允许中断/中断程序LD SM0.0 DTCH 10 /关闭定时中断10XMT VB100, 0 /回送接收到的数据ATCH INT_0, 8 /准备接收下一帧报文需要注意的是,如果使用PC/PPI电缆,在S7-200 CPU的用户程序中应考虑电缆的切换时间。从接收到请求报文后到发送响应报文的延迟时间和再次发出请求报文的延迟时间都必须大于等于电缆的切换时间。通信波特率为9600bps和19200bps时,切换时间分别为2ms和1ms。4 上位机通信编程在Windows环境下开发与工业PLC通信的软件,利用VisualC+6.0的通信控件MSComm控件编写软件程序。该通信控件提供了使用RS232开发串行通信软件的细则,使用事件驱动或查询方式来解决开发通信软件中遇到的问题。在当前工程中插入Mscomm控件时,如果是基于对话框或视图类是基于CformView的应用程序,加入方法十分简单。如果不是,就需要创建一个Mscomm控件。方法与普通的ActiveX控件的创建类似。首先在MainFrm.h中添加声明语句:protected: CMSComm m_myCom;然后在MainFrm.cpp:OnCreate()函数中添加下列创建控件的语句:DWORD style=WS_VISIBLE|WS_CHILD;If(!m_myCom.Create(NULL,style,CRect(0,0,0,0),this,ID_COMMCTRL)TRACE0(Failed to create OLE Communications Control );Return -1;开发通信程序部分代码如下:/ 首先初始化通信端口 m_myCom.SetCommPort(1); /选择COM1m_myCom.SetInBufferSize(1024); /设置输入缓冲/区的大小m_myCom.SetOutBufferSize(512); /设置输出缓/冲区的大小m_myCom.SetInputMode(1); /设置数据获取方式m_myCom.SetSettings(19200,N,8,1); /设置通/信参数m_myCom.SetInputLen(0); /设置读取方式if(!m_myCom.GetPortOpen() ); m_myCom.SetPortOpen (TRUE); /打开串口收发数据时我们一般习惯于使用字符串形式(数组形式),而读写函数GetInput()和SetOutput()都要使用VARIANT类型(查阅MSDN可知,IDispatch:Invoke()的参数和返回值作为VARIANT对象处理),可以用BSTR表示字符串(包含宽字符),为了解决系统不支持宽字符的问题,使用了CByteArray,部分程序如下:/ 接收事件void CCommDlg:OnComm()int iCount;if(m_myCom.GetCommEvent()=2) /接收事件,在缓冲区内数据被移走前持续产生iCount=m_myCom.GetInBufferCount(); /接收到字符数目m_myCom.SetInputLen(iCount);if(iCount0)COleVariant vRec=m_myCom.GetInput();/读取字符vRec. ChangeType (VT_BSTR); m_response = vRec.bstrVal;/把接收到的变量赋给应答帧文本框变量 UpdateData (FALSE); /数据处理/ 发送事件 void CCommDlg:OnSend () / 准备需要发送的命令,放在SendData中CByteArray array; array.RemoveAll();array.SetSize(iCount);for(int i=0;i);m_myCom.SetOutput (COleVariant (array); ) 5 结束语西门子S7-200型PLC有很强的通信功能,在硬件配置与安装上,建议系统交流电源使用双层隔离,输入信号光电隔离,提高信噪比,远离强电布线,模拟量信号和数字信号采用屏蔽线传送,采用可靠接地等措施,能有效地消除或减弱外界信号干扰。在软件设计与编程上,加入抗干扰模块,如采用软件滤波技术,对一些重要模拟量参数进行延时判定,并进一步确认报警;在通讯软件设计中采用多种校验措施等都能大大提高系统的抗干扰性能。采用本方法设计的通讯程序简单实用、维护和扩充方便,对同类系统的设计与实现有一定的启发。1.建立项目2.在项目中插入MSComm控件3.利用ClassWizard定义CMSComm类控制变量4.在对话框中添加控件5.添加串口事件消息处理函数OnComm()6.打开和设置串口参数7.发送数据8.发送十六进制字符9.在接收框中以十六进制显示10.如何设置自动发送11.什么是VARIANT数据类型?如何使用VARIANT数据类型? 这是串口调试助手源程序及编程详细过程(一)的续篇,首先谢谢朋友们的支持与鼓励。8.发送十六进制字符 在主对话框中加入一个复选接钮,ID为IDC_CHECK_HEXSEND Caption: 十六进制发送,再利用ClassWizard为其添加控制变量:m_ctrlHexSend; 在ClassView中为SCommTestDlg类添加以下两个PUBLIC成员函数,并输入相应代码;/由于这个转换函数的格式限制,在发送框中的十六制字符应该每两个字符之间插入一个空隔/如:A1 23 45 0B 00 29/CByteArray是一个动态字节数组,可参看MSDN帮助int CSCommTestDlg:String2Hex(CString str, CByteArray &senddata)int hexdata,lowhexdata;int hexdatalen=0;int len=str.GetLength();senddata.SetSize(len/2);for(int i=0;i=len)break;lstr=stri;hexdata=ConvertHexChar(hstr);lowhexdata=ConvertHexChar(lstr);if(hexdata=16)|(lowhexdata=16)break;elsehexdata=hexdata*16+lowhexdata;i+;senddatahexdatalen=(char)hexdata;hexdatalen+;senddata.SetSize(hexdatalen);return hexdatalen;/这是一个将字符转换为相应的十六进制值的函数/好多C语言书上都可以找到/功能:若是在0-F之间的字符,则转换为相应的十六进制字符,否则返回-1char CSCommTestDlg:ConvertHexChar(char ch)if(ch=0)&(ch=A)&(ch=a)&(ch=f)return ch-a+10;else return (-1); 再将CSCommTestDlg:OnButtonManualsend()修改成以下形式:void CSCommTestDlg:OnButtonManualsend()/ TODO: Add your control notification handler code hereUpdateData(TRUE); /读取编辑框内容if(m_ctrlHexSend.GetCheck()CByteArray hexdata;int len=String2Hex(m_strTXData,hexdata); /此处返回的len可以用于计算发送了多少个十六进制数m_ctrlComm.SetOutput(COleVariant(hexdata); /发送十六进制数据elsem_ctrlComm.SetOutput(COleVariant(m_strTXData);/发送ASCII字符数据现在,你先将串口线接好并打开串口调试助手V2.1,选上以十六制显示,设置好相应串口,然后运行我们这个程序,在发送框中输入00 01 02 03 A1 CC等十六进制字符,并选上以十六进制发送,单击手动发送,在串口调试助手的接收框中应该可以看到00 01 02 03 A1 CC了。9.在接收框中以十六进制显示 这就容易多了:在主对话框中加入一个复选接钮,IDC_CHECK_HEXDISPLAY Caption: 十六进制显示,再利用ClassWizard为其添加控制变量:m_ctrlHexDiaplay。 然后修改CSCommTestDlg:OnComm()函数:void CSCommTestDlg:OnComm()/ TODO: Add your control notification handler code hereVARIANT variant_inp;COleSafeArray safearray_inp;LONG len,k;BYTE rxdata204

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论