作者:紫逸石 | 来源:互联网 | 2023-09-08 16:10
MSComm控件串口通信开发 1. MSComm控件的处理流程 MSComm在适当时候引发控件的处理流程主要分为两步:
1)通过事件驱动处理连接端口通信 在收发一个字符或者生一个事件时,不管是产生一个事件还是错误(这取决于CommEvent的属性值),都会执行控件的事件程序OnComm()。我们需要预先将处理事件或者错误的逻辑代码输入在控件事件区。
2)检查CommEvent的属性值 通过检查CommEvent的值来判断是事件还是错误,并根据触发事件时CommEvent不同的值编写不同的分支来处理对应事件。对于处理串口收发数据,最常用的事件属性值有comEvSend(发送数据) 和 comEvReceive(接收数据) 。
2.上位机和下位机之间连接 在早期,上位机和下位机之间经常使用的通信方式是RS-232的串行通信控制。直到今天,在USB通信普遍使用的方式下,PC端的应用软件依然是应用RS-232通信,保护了原有的软件开发投入。本次实验用的接口是较为常见USB 接口,板子上采用了USB转串口芯片进行信号的转换。
局部功能 1.检查当前可用的串口 上篇文章中讲到,VB6.0中MSComm控件的端口号仅限于1~16,并且大多数情况下用于插拔的USB都不是一个固定值,因此,我们首先需要获取PC机当前状态下可用于连接的串口端口号。
代码如下:
On Error Resume Next Dim I As Integer For I &#61; 1 To 16 Step 1 MSComm1.CommPort &#61; IMSComm1.PortOpen &#61; True If Err.Number <> comPortInvalid Then Combo1.AddItem "COM" & I End If MSComm1.PortOpen &#61; False Next Combo1.Text &#61; Combo1.List( 0 )
效果如下图&#xff1a; 相关代码 &#xff1a;
(1)On Error Resume Next 加上这句语句后&#xff0c;若当程序运行到某处出错时&#xff0c;不会中断&#xff0c;而是会存储出错信息到某处&#xff0c;然后继续执行后面的代码 。在这里使用它的原因是&#xff0c;如果当时上位机并没有相应的串口设备注册&#xff0c;那么当我们企图打开一个串口时&#xff0c;会因错误中断而导致程序无法进行下去。加上这句话后&#xff0c;再结合MSComm1.PortOpen属性判断串口是否正确打开&#xff0c;以便于进行接下来的不同操作。
(2)Err.Number <> comPortInvalid 这句代码配合前面的On Error Resume Next 使用。当发现打开串口出错时&#xff0c;继续执行后续代码&#xff0c;但这时已经将错误码存储于Err.Number 之中&#xff0c;检查其是否和comPortInvalid 相等&#xff0c;可判断该端口是否可用于串行通信。是&#xff0c;则将其加入下拉列表中。
2.串口打开按钮 在找到对应的可打开串口后&#xff0c;选择其中一个串口打开&#xff0c;这时候&#xff0c;也需要检查串口是否能够正确打开&#xff0c;如果出错&#xff0c;则对用户进行提示。针对打开按钮的单击操作如下&#xff1a;
On Error Resume Next ’打开出错时不中断If MSComm1.PortOpen &#61; False Then MSComm1.CommPort &#61; Right ( Combo1.Text , Len ( Combo1.Text ) - 3 ) MSComm1.PortOpen &#61; True If Err.Number &#61; comPortInvalid Then MsgBox "无法打开串口&#xff0c;请检查串口是否有效"&#xff0c; vbOKOnly&#xff0c; "串口打开错误"Else Command1.Caption &#61; "关闭"End IF Else MSComm1.PortOpen &#61; False Command1.Caption &#61; "打开"End If
在上述代码中&#xff0c;如果不能 正确打开串口&#xff0c;会弹出消息框(MsgBox)报错&#xff0c;如下图&#xff1a;
3.串口通信的设置 在使用串口通信前&#xff0c;或者说打开串口前 &#xff0c;应该对串口通信参数进行基础配置。这些配置参数可以在添加控件时的属性栏中进行编辑&#xff0c;也可以独自在代码中列出&#xff0c;如下&#xff1a;
MSComm1.Settings &#61; "9600 &#xff0c;o&#xff0c;8 &#xff0c;2 " MSComm1.InBufferSize &#61; 1024 MSComm1.OutBufferSize &#61; 512 MSComm1.InputLen &#61; 32 MSComm1.InputMode &#61; comInputModeBinary MSComm1.RThreshold &#61; 9 MSComm1.RTSEnable &#61; True MSComm1.SThreshold &#61; 0
4.接收数据帧展示(十六进制) 我们可以采用一个Text文本框进行接收数据的显示&#xff0c;将二进制字节数据转化为十六进制数据并以空格分隔显示在文本框当中。 1&#xff09;对于文本框的设置 对于文本框&#xff0c;我们希望其可以将每一帧数据进行滚动显示&#xff0c;并且每次都显示最后一帧数据&#xff0c;于是进行如下设置&#xff1a;
Text28.MultiLine &#61; True Text28.ScrollBars &#61; 2 Text28.SelStart &#61; Len ( Text28.Text )
2&#xff09;进制转换代码
Public Function ReceiveToHex( bytInput( ) As Byte ) As String Dim I As Integer Dim strData As String For I &#61; 0 To UBound ( bytInput) If Len ( Hex( bytInput( I) ) ) &#61; 1 Then strData &#61; strData & "0" & Hex( bytInput( I) ) & " " Else strData &#61; strData & Hex( bytInput( I) ) & " " End If Next ReceiveToHex &#61; strData & vbCrLf End Function
效果图如图所示&#xff1a;
5.发送数据设置 数据的发送使用了一个50ms的定时器&#xff0c;对发送数组进行内容填充更新&#xff0c;然后将该数组的值送入发送缓冲区。在定时器执行过程中有如下代码&#xff1a;
PackTranData MSComm1.Output &#61; bytOutput
**注&#xff1a;**由于数据发送缓冲区的数据填充过程并不是每次都从头开始存放数据&#xff0c;而是采用游标移动添加数据&#xff0c;所以每次数据帧在传送字节流的具体位置并不固定&#xff0c;比如&#xff1a;这次发送的数据是&#xff1a;
00 00 5 A A5 06 00 00 00 00 00 00 00 00 00 00 00 00 00
下一次的数据很可能是&#xff1a;
00 00 00 00 00 00 00 00 00 00 5 A A5 06 00 00 00 00 00
所以这时候&#xff0c;在下位机的接收端找到对应的数据帧十分重要&#xff0c;帧头和帧尾的重要性凸显。