【急求】mfc 实现串口编程的源代码

【急求】mfc 实现串口编程的源代码,第1张

1.建立项目:打开VC++6.0,建立一个基于对滑薯话框的MFC应用程序SCommTest(与我源代码一致,等会你会方便一点);

2.在项目中插入MSComm控件 选择Project菜单下Add To Project子菜单中的 Components and Controls…选项,在d出的对话框中双击Registered ActiveX Controls项(稍等一会,这个过程较慢),则所有注册过的ActiveX控件出现在列表框中。 选择Microsoft Communications Control, version 6.0,,单击Insert按钮将它插入到我们的Project中来,接受缺省的选项。(如果你在控件列表中看不到Microsoft Communications Control, version 6.0,那可能是你在安装VC6时没有把ActiveX一项选上,重新安装VC6,选上ActiveX就可以了),

这时在ClassView视窗中就可以看到CMSComm类了,(注意:此类在ClassWizard中看不到,重构clw文件也一样),并且在控件工具栏Controls中出现了电话图标(如图1所示),现在要做的是用鼠标将此图标拖到对话框中,程序运行后,这个图标是看不到的。

3.利用ClassWizard定义CMSComm类控制对象打开ClassWizard->Member Viariables选项卡,选择CSCommTestDlg类,为IDC_MSCOMM1添加控制变量:m_ctrlComm,这时你可以看一看,在对话框头文件中自动加入了//{{AFX_INCLUDES()#i nclude "mscomm.h"//}}AFX_INCLUDES (这时运行程序,如果有错,那就再从头开始)。

4.在对话框中添加控件向主对话框中添加两个编辑框,一个用于接收显示数据ID为IDC_EDIT_RXDATA,另一个用于输入发送数据,ID为IDC_EDIT_TXDATA,再添加一个按钮,功能是按一次就把发送编辑框中的内容发送一次,将其ID设为IDC_BUTTON_MANUALSEND。别忘记了将接收编辑框的Properties->Styles中把Miltiline和Vertical Scroll属性选上,发送编辑框若你想输入多行文字,也可选上Miltiline。

再打开ClassWizard->Member Viariables选项卡,选择CSCommTestDlg类, 为IDC_EDIT_RXDATA添加CString变量m_strRXData, 为IDC_EDIT_TXDATA添加CString变量m_strTXData。说明: m_strRXData和m_strTXData分别用来放入接收和发送的字符数据。

5.添加串口事件消息处理函数OnComm() 打开ClassWizard->Message Maps,选择类CSCommTestDlg,选择IDC_MSCOMM1,双击消息OnComm,将d出的对话框中将函斗友数名改为OnComm,(好记而已)OK。

这个函数是用来处理串口消息事件的,信销者如每当串口接收到数据,就会产生一个串口接收数据缓冲区中有字符的消息事件,我们刚才添加的函数就会执行,我们在OnComm()函数加入相应的处理代码就能实现自已想要的功能了。请你在函数中加入如下代码:

void CSCommTestDlg::OnComm()

{

// TODO: Add your control notification handler code here

VARIANT variant_inp

COleSafeArray safearray_inp

LONG len,k

BYTE rxdata[2048]//设置BYTE数组 An 8-bit integerthat is not signed.

CString strtemp

if(m_ctrlComm.GetCommEvent()==2) //事件值为2表示接收缓冲区内有字符

{ ////////以下你可以根据自己的通信协议加入处理代码

variant_inp=m_ctrlComm.GetInput()//读缓冲区

safearray_inp=variant_inp//VARIANT型变量转换为ColeSafeArray型变量

len=safearray_inp.GetOneDimSize()//得到有效数据长度

for(k=0k<lenk++)

safearray_inp.GetElement(&k,rxdata+k)//转换为BYTE型数组

for(k=0k<lenk++) //将数组转换为Cstring型变量

{

BYTE bt=*(char*)(rxdata+k)//字符型

strtemp.Format("%c",bt)//将字符送入临时变量strtemp存放

m_strRXData+=strtemp//加入接收编辑框对应字符串

}

}

UpdateData(FALSE)//更新编辑框内容

}

到目前为止还不能在接收编辑框中看到数据,因为我们还没有打开串口,但运行程序不应该有任何错误,不然,你肯定哪儿没看仔细,因为我是打开VC6对照着做一步写一行的,运行试试。没错吧?那么做下一步:

6.打开串口和设置串口参数你可以在你需要的时候打开串口,例如在程序中做一个开始按钮,在该按钮的处理函数中打开串口。现在我们在主对话框的CSCommTestDlg::OnInitDialog()打开串口,加入如下代码:

// TODO: Add extra initialization here

if(m_ctrlComm.GetPortOpen())

m_ctrlComm.SetPortOpen(FALSE)

m_ctrlComm.SetCommPort(1)//选择com1

if( !m_ctrlComm.GetPortOpen())

m_ctrlComm.SetPortOpen(TRUE)//打开串口

else

AfxMessageBox("cannot open serial port")

m_ctrlComm.SetSettings("9600,n,8,1")//波特率9600,无校验,8个数据位,1个停止位

m_ctrlComm.SetInputMode(1)//1:表示以二进制方式检取数据

m_ctrlComm.SetRThreshold(1)

//参数1表示每当串口接收缓冲区中有多于或等于1个字符时将引发一个接收数据的OnComm事件

m_ctrlComm.SetInputLen(0)//设置当前接收区数据长度为0

m_ctrlComm.GetInput()//先预读缓冲区以清除残留数据

现在你可以试试程序了,将串口线接好后,打开串口调试助手,并将串口设在com2,选上自动发送,也可以等会手动发送。再执行你编写的程序,接收框里应该有数据显示了。

7.发送数据先为发送按钮添加一个单击消息即BN_CLICKED处理函数,打开ClassWizard->Message Maps,选择类CSCommTestDlg,选择IDC_BUTTON_MANUALSEND,双击BN_CLICKED添加OnButtonManualsend()函数,并在函数中添加如下代码:

void CSCommTestDlg::OnButtonManualsend()

{

// TODO: Add your control notification handler code here

UpdateData(TRUE)//读取编辑框内容

m_ctrlComm.SetOutput(COleVariant(m_strTXData))//发送数据

}

运行程序,在发送编辑框中随意输入点什么,单击发送按钮,我们通过把RS232的2.3两口短接,在一台电脑上显示串口的收发数据!

最后说明一下,由于用到VC控件,在没有安装VC的计算机上运行时要从VC中把mscomm32.ocx、msvcrt.dll、mfc42.dll拷到Windows目录下的System子目录中(win2000为System32)并再进行注册设置

添加一个mscomm 控件里面有,然后设置它的波特率,波特率和友前瞎下位机的波特率要一致。然后就用

Write和Read方法读写数据就行了。

推荐关于MSCOMM控件的一些说明

VB5.0/6.的MSComm通信控件提供了一系列标准通信命令的接口,它允许建立串口连接,可以连接到其他通信设备(如Modem).

还可以发送命令、进行数据交换以及监视和响应在通信过程中可能发生的各种错误和事件,从而可以用它创建全双工 、事件驱

动的、高效实用的通信程序。但在实际通信软件设计过程中,MSComm控件并非像想像中那样完美和容易控制.特别是在中文Wln

95/98下通信时更会出现问题。下面就从基础开始介绍,然后逐步讨沦MSComm控件在编程中出现的问题以及编程技巧。

一、用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 1002Clear To Send Timeout。在发送字符时,在系统指定的事1件内,CTS(Clear To Send)线是低电平

ComEventDSRTO 1003Data Set Ready Timeout。在发送字符时,在系统指定的事件内,DSR(Data Set Ready)线是低电平

ComEventFrame 1004数据帧错误。硬件检测到一个数据帧错误

ComEventOverrun 1006端口溢出。硬件中的字符尚未读,下一个字符又到达,并且丢失

ComEventCDTO1007Carrier 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 3CTS(Clear To Send)线改变

ComEvDSR 4DSR(Data Set Ready)线改变。当DSR从1到0改变时,该事件发生

ComEvCD5CD(Carrier Detect)线改变ComEvRing6检测到响铃信号。一些URAT(Universal AsynchronousReciver-

-Transmitters,通用异步收发器)不支持该事件

ComEvEOF 7收到了EOF字符(ASCII字符26)

Error消息(MSComm控件)下表列出了MSComm控件可捕获的错误消息:

常量 值 描述

ComInvalidPropertyValue380 无效的属性值

ComSetNotSupported 383 属性只读

ComGetNotSupported 394 属性只读

ComPortOpen 8000 端口打开时该存在无效

8001 超时设置必须比0值大

ComPortInvalid8002 无效的端口号

8003 属性只在运行时有效

8004 属性在运行时是只读的

ComPortAleadyOpen 8005 端口已经打开

8006 设备标识符无效或不支持

8007 不支持设备的波特率

8008 指定的字节大小无效

8009 缺省参数错误

8010 硬件不可用(被其他设备锁住)

8011 函数不能分配队列

ComNoOpen 8012 设备没有打开

8013 设备已经打开

8014 不能使用通信通知

ComSetCommStateFailed 8015 不能设置通信状态

8016 不能设置通信事件屏蔽

ComPortNotOpen8018 该存在只在端口打开是有效

8019 设备忙

ComReadError 8020 通信设备读错误

ComDCBError 8021 检取端口设备控制块时出现内部错误

搞清楚以上基本属性后,就可以开始编写通信许程序了。在VB5.0/6.0中新建一个工程文件。添加Microsoft Comm Control 5.0组

件,在简体Form1中加入Command命令按钮并取名为CmdTest,MSComm控件取名为MSComm1,加入如下程序代码。

Private Sub cmdTestClick ( )'打开串口

MSComml.CommPort =2 '设定Com2

If MSComml.PortOpen = False Then

MSComm1.Settings = "9600,n,8,1" '9600波特率,无校验,8位数据位,1位停止位

MSComm1.PortOpen = True '打开串口

End if

MSComm1.OutBufferCount = 0 '清空发送缓冲区

MSComm1.InBufferCount = 0 '滑空接收缓冲区

'发送字符数据时注意必须用回车符(vbcr)结束

MSComm1.Output="This is a qood book ! " &vbCr

'泼打电话号码或发送AT命令

MSComm1.Output = "ATDT 05778191898 , &vbCr

'发送字符数组数据时注意ByteArray必须事先定义赋值

Dim ByteArray as byte( )

'定义动态数组

ReDim ByteArray(1)

'重定义数组大小

ByteArray ( 0 ) =0

ByteArray ( 1 ) = 1

MSComm1.Output = ByteArray

End Sub

private Sub MScommEvent( )

Select Case MSComm1.CommEvent

Case comEvReceive

Dim Buffer As Variant

MSComm1.InputLen = 0

'接收二进制数据

MSComm1.InputMode= ComInputModeBinary

Buffer=MSComm1.Input

'接收字符数据

MSComm1.InputMode=comInputModeText

Buffer = MSComml.Input

Case else

End Select

End sub

( 程序1)

二、中文Win 95/98下的通信问题与解决方法

1.接收的数据少于发送的数据

如果通过MSComm控件一次性传送较多的二进制数据,那么,很可能收到的数据不足。例如在设置为24oobps传输率的情况下,

一次性可以传输2048个字符数据 那么在大多数情况下。一次只能收到1200个字符左右,这址出为新版的MSComm32.OCX中存在一

个影响传输二进制数据的臭虫(bug).注意这不是特性。

32位Windows API函数(以下简称API)使用了几个用COMMTIMEOUTS结构表示的限时变量,WriteTotalTimeOutConstant 即是其

中的一个,它被Windows内部设定为5000(即5秒),这个常量决定了在通信驱动程序停止传输之前花费在发送缓冲区中数据的时间

的长短,5秒钟意味着通信速度为1200bps情况下仅能发送600个字符,24oobps情况下仅能发送1200个左右的字符。事实上,在一个

缓冲区内一次性发送更多的数据是非常可能的。这个bug同样也能引发问题,甚至在高速串口门通信情况下,即使系统在使用流控

制,无论丛软件流(Xon/XofI)还是硬件流(CTS/RTS)。假如数据在发送缓冲区中时,流控制停止了传输,如果停止时间超过5

秒钟.则数据就会丢失。在某些环境下,5秒钟可能相当短.不过也不必担心, VB 5.0/6.0版本的MSComm控件有一个新增的重要的

属性称为CommID, CommID指的是当串口被打开时,被API所调用的串口句柄或称标志,这也意味着能利用API接口函数去修改这个

常量。每次串口关闭后,Windows会自动将之恢复为5000,所以,每次打开串口后需要重斩设定以下API声明,其代码见下程序。

Type COMMTIMEOUTS

ReadIntervalTimeout As Long

ReadTotalTimeoutMultiplier As Long

ReadTotalTimeoutConstant As Long

WriteTotalTimeoutMultiplier As Long

WriteTotalTimeoutConstant As Long

End Type

Declare Function SetCommTimeouts Lib "Kernel32"

(BYVal hFile As Long, lpComm TimeoutsAs COMMTIMEOUTS) As Long

Declare Function GetCommTimeouts Lib "Kernel32"

(ByVal hFile As Long, lpCommTimeouts As COMMTIMEOUTS) As Long

Dim timeouts As COMMEOUTS

Dim Ret As Long

If Comm1.PortOpen = False Then

Comm1.PortOpen = True

End if

Ret=GetCommTimeouts ( Comm1.CommID , timeouts )

'Set some default timeouts

timeOuts.ReadIntervalTimeout = 1

timeouts.ReadTotalTimeoutMultiplier =1

timeouts.ReadTotalTimeoutConstant =1

timeouts.WriteTotalTimeoutMultiplier =1

timeouts.WriteTotalTimeoutConstant=

( Comm1.OutBufferSize\Val(Comm1.Settings))*10000+1000

Ret=SetCommTimeouts( Comm1.CommID , timeouts )

( 程序2)

2.如何发送大于128的字符数据

在通信程序中,以单字符方式逐个发送数据时,每一个数据范围 0-255(即十六进制的00-FF)。在单字符版本的英文Win95或

DOS版的BASIC程序中,只需要将相应的数据转换成相应的字符发送到通信端口即可。但在中文Win95/98下却行不通,假设在中文

Win95/98下运行以下程序:

Dim i

For i=0 to 255

MSComm1.Output=chr(i)

Next i

希望在接收端得到预期的0-255之间的数据,结果却是:前129个数据接收正确,为0-128,后面127个数据为126个0和一个255,

造成这种给果的原因在于中文Windows使用的是双字节字符集(DBCS)系统。DBCS系统使用0-128之间的数字表示ASCII字符,大于

128的数字仅作为前导字符,它只是显示是一个非拉丁语系的字符,而并不代表实际意义。上述程序在调用CHR()函数时用到了

DBCS字符集,冈此产生了此类错误。那么,如何发送人于128的数据呢?答案是使用字符数组,将以上程序改为:

Dim cc(255) As Byte

For i = 0 To 255

cc(i) = i

Next i

MSComm1.Output = cc

Do

DoEvents

Loop Until MSComm1.OutBufferCount = 0

'接收过程 MSComm1_OnComm()

Select Case MSComm1.CommEvent

Case comEvReceive

Dim Buffer As Variant, b1,i

MSComm1.InputMode=comInputModeBinery

MSComm1.InputLen = 0

Buffer = MSComm1.Input

For i=LBound (Buffer) To UBound (Buffer )

Debug.Print Buffer ( i )

Next i

Case . . . . .

3.如何发送0字符(00H,NULL)

在VisuaI C++中使用串口控件发送0字符有些麻烦,但在VB5.0/6.0中只要注意以下两点即可:

(1)设置MSComm控件的属性 NullDiscard=False;。

(2)使用二进制接收,即用 MSComm1.InputMode=ComInputModeBinary便可以解决问题;

4.如何发送递中文字符串(DBcS字符)

VB5.0/6.0的各种参考书上均指明MSComm通信控件不能发送或接收双字节字符集系统DBCS)的二进制数据,这对于我国及亚洲一些

使用DBCS字符集的国家不能不说是一大人遗憾。但是我在实践中发现,用MSComm控件也可以发送中文字符,具体方法有以下两种:

(1)直接发送

直接发送即把中文字符等同于英文字符。如:MSComm1.Intput= " 这是一行中文数据!" ,但这种方法发送的中文数据不能太

长,发送缓冲区和接收缓冲区的大小需设定为中文字符的两倍以上,而且发送与接收系统所处的 *** 作系统版本最好要一致,否则会

出现接收或发送缓冲区溢出之类的错误。这种方法时用于一般要求不太高的场合。

(2)间接发送

在发送端将汉字或字符转换为机器内码或区位码数据数组,然后将咏转换后的数据发送到串口,在接收端接收到数据后,按照

相反的顺序得到的数据转换为相应的汉字或字符,在转换过程中.要用到位运算,如取得汉字的内码后需要将高字节和低字节分开,

而VB5.0/6.0中并没有提供此类函数,以下是求整数高、低字节的函数。

Public Function HiByte(a As Integer )

Dim b

b= a And &HFF00

b = b / 256

If b<0 Then b = b + 256

HiByte = b

End Function

Public Function LowByte(a As Integ`er)

Dim b

b = a And &HFF

LowByte = b

End Function

5.如何用单机进行通信测试

通常在写好了通信程序后需要两台PC或一台Pc、一台单片机.将通信口连接后进行测试,但很多时侯因条件限制仅有单台PC机,

测试项目很简单,那么能否测试呢?当然可以,而且方法也很简单。对于九针的串口,找一个废弃的串口鼠标,剥外鼠标线,将连

接2、3针的线对接即可;对于25针的串口,找一枚曲别针(最好有塑料外套的)将它扯直,剥削去两头的塑料后在两头各弯一个圆

圈,中间对忻后直接套接在串口的2、3针上即可。如果但心不够安全,则可以将5针按地。

'''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''

关于mscomm的用法,提高篇......[mgwmj]?

MSCOMM控件是个好东西,如果您能够充分了解他,他会为您衷心的效劳。

大致看了一下下午有关讨论MSCOMM的话题,觉得有必要说说我的心得,我一般只做硬件,没有系统的学过软件,只是业余时间

学学用用,多少掌握了一点,也在此拿出来玩玩,不知有错没有,我可是以为我已经做的很好了^_^

这是一个VB通用串口事件驱动接收程序。一次性接收一个数据包,数据包可以为任意字节,保证不会丢失一个数据!

Private Sub MSComm_OnComm()

Dim S() As Byte

Dim SS(1024) As Byte

Static N As Long

Static T As Variant

If (MSComm.CommEvent = comEvReceive) Then

S = MSComm.Input '只要有数据就收进来,哪怕只是一个

If (Timer - T >0.01) Then'间隔10MS以上就认为是一个新的包

text1="" 'text1用于搜集和显示接收(HEX格式)

N = 0

End If

T = Timer

For i = 0 To UBound(S) '一个数据包可能产生若干个oncomm事件

Text1.Text = Text1.Text &Right("0" &Hex(S(i)) &"H", 3) + " "

SS(N+i)=S(i) '接收数据包缓存于SS()

N=N+UBound(S)

Next i

End If

End Sub


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/yw/12414210.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-25
下一篇 2023-05-25

发表评论

登录后才能评论

评论列表(0条)

保存