vb mscomm 接收不到数据

tornado_dai 2013-07-28 11:18:26

Private Sub Form_Load()
Dim i As Integer
If MSComm1.PortOpen = True Then
MSComm1.PortOpen = False
End If '先判断串口是否关闭,如果打开则关闭
MSComm1.CommPort = 5 'COM端口
MSComm1.Settings = "115200,n,8,1"
MSComm1.InputLen = 0 '读取接收缓冲区的所有字符
MSComm1.InputMode = comInputModeBinary '采用二进制传输
MSComm1.InBufferSize = 1024 '接受缓冲区为1024字节
MSComm1.OutBufferSize = 2048 '发送缓冲区位2048字节
MSComm1.InBufferCount = 0 '清空接受缓冲区
MSComm1.OutBufferCount = 0 '清空传输缓冲区
MSComm1.SThreshold = 0 '禁止发送事件
MSComm1.RThreshold = 1 '产生MSComm事件
Command2.Enabled = False '初始化时设置“关闭”按钮不可用
End Sub

Private Sub MSComm1_OnComm()

Dim inbyte() As Byte '存储接收串口数据
Dim buffer As String '存储处理后的串口数据
Dim lenbuffer As Integer '存储处理后的串口数据长度
Dim i As Integer '设置循环变量

Select Case MSComm1.CommEvent
Case comEvReceive
inbyte = MSComm1.Input
For i = LBound(inbyte) To UBound(inbyte)

If Len(Hex(inbyte(i))) = 1 Then
buffer = buffer + "0" + Hex(inbyte(i)) + Chr(32)
Else
buffer = buffer + Hex(inbyte(i)) + Chr(32)
End If

lenbuffer = Len(Replace(Trim(buffer), " ", ""))

Select Case lenbuffer
Case Is = 14
If CByte("&H" & (Mid(Replace(buffer, " ", ""), 1, 2))) = &HAA _
And "&H" & (Mid(Replace(buffer, " ", ""), 14 - 4 + 1, 2)) = &H42 _
And "&H" & (Mid(Replace(buffer, " ", ""), 14 - 2 + 1, 2)) = &H42 Then
If "&H" & (Mid(Replace(buffer, " ", ""), 14 - 12 + 1, 2)) = &H55 Then

If "&H" & (Mid(Replace(buffer, " ", ""), 14 - 10 + 1, 2)) = &H43 _
And "&H" & (Mid(Replace(buffer, " ", ""), 14 - 8 + 1, 2)) = &H47 _
And "&H" & (Mid(Replace(buffer, " ", ""), 14 - 6 + 1, 2)) = &H8A Then
Text2.Text = "测试通过"
End If

End If
End If

case is = 18
....... 省略

End Select
Next i
End Sub


想问下,为什么上位机接收不到数据???
...全文
520 24 打赏 收藏 转发到动态 举报
写回复
用AI写文章
24 条回复
切换为时间正序
请发表友善的回复…
发表回复
tornado_dai 2013-08-06
  • 打赏
  • 举报
回复
引用 23 楼 zhao4zhong1 的回复:
推荐使用portmon软件辅助串口通讯调试。
不错,好用
赵4老师 2013-08-02
  • 打赏
  • 举报
回复
推荐使用portmon软件辅助串口通讯调试。
tornado_dai 2013-08-01
  • 打赏
  • 举报
回复
引用 21 楼 myjian 的回复:
我一般的做法是,收到数据先检查长度,再与已存储的长度相加,看看是不是已经达到我要求的长度. 如果达到或超过,就进入处理流程. 处理流程里会找到有效数据的头与尾,并清空尾部数据以前的数据,保证未处理的数据是在缓冲区开头. 如此周而复始.
多谢分享
嗷嗷叫的老马 2013-07-31
  • 打赏
  • 举报
回复
我一般的做法是,收到数据先检查长度,再与已存储的长度相加,看看是不是已经达到我要求的长度. 如果达到或超过,就进入处理流程. 处理流程里会找到有效数据的头与尾,并清空尾部数据以前的数据,保证未处理的数据是在缓冲区开头. 如此周而复始.
zdingyun 2013-07-30
  • 打赏
  • 举报
回复
引用 19 楼 zdingyun 的回复:
接收数据是固定字节长度的话,由于可以设置MSComm1.RThreshold为固定字节长度,buffer定义于为过程变量,接收代码照你原先的是没问题的。 而接收不定长的字节,只能设置MSComm1.RThreshold = 1,而COM口硬件原因,大于8字节以上的数据到达COM口会多次触发OnComm事件,buffer定义于为过程变量会导致接收字节不完整有丢数据现象,所以buffer定义于为全局变量。
大于8字节以上是我这儿COM口硬件测试结果,LZ接收”aa 55 43 47 8a 42 42“7字节有丢字节现象,我这儿没遇见过。或许LZ的环境电磁干扰或上下位机波特率差异等因素导致。
zdingyun 2013-07-30
  • 打赏
  • 举报
回复
接收数据是固定字节长度的话,由于可以设置MSComm1.RThreshold为固定字节长度,buffer定义于为过程变量,接收代码照你原先的是没问题的。 而接收不定长的字节,只能设置MSComm1.RThreshold = 1,而COM口硬件原因,大于8字节以上的数据到达COM口会多次触发OnComm事件,buffer定义于为过程变量会导致接收字节不完整有丢数据现象,所以buffer定义于为全局变量。
tornado_dai 2013-07-30
  • 打赏
  • 举报
回复
多谢zdingyun , 将buffer定于为全局变量的方法可以接收到完整数据,接收到下一组数据时,之前的数据还在,需做一下清空处理; 可不可以讲解一下我之前接收数据丢失以及用这钟方法的缘由?
tornado_dai 2013-07-29
  • 打赏
  • 举报
回复
Private Sub Command1_Click() '打开串口 On Error GoTo Err MSComm1.CommPort = CInt(Mid(C1.Text, 4, 1)) '选择端口号 If MSComm1.PortOpen = False Then MSComm1.PortOpen = True Text1.Text = "串口打开" Command1.Enabled = False '设置“打开”按钮变灰 Command2.Enabled = True Exit Sub End If Err: MsgBox "端口被占用或无", vbCritical, "错误警告" End Sub 串口打开代码如上,上面的按你说的改过试了,还是不能接收到数据。。。
zdingyun 2013-07-29
  • 打赏
  • 举报
回复
从代码看COM口没打开。 OnComm事件代码缺 End Select Next i位置错误

Private Sub Form_Load()
    Dim i As Integer
    If MSComm1.PortOpen = True Then
        MSComm1.PortOpen = False
    End If
    MSComm1.CommPort = 1
    MSComm1.Settings = "115200,n,8,1"
    MSComm1.InputLen = 0
    MSComm1.InputMode = comInputModeBinary
    MSComm1.InBufferSize = 1024
    MSComm1.OutBufferSize = 2048
    MSComm1.InBufferCount = 0
    MSComm1.OutBufferCount = 0
    MSComm1.SThreshold = 0
    MSComm1.RThreshold = 1
    MSComm1.PortOpen = True '打开COM口
    Command2.Enabled = False
End Sub

Private Sub MSComm1_OnComm()
    Dim inbyte() As Byte
    Dim buffer As String
    Dim lenbuffer As Integer
    Dim i As Integer
    Select Case MSComm1.CommEvent
        Case comEvReceive
            inbyte = MSComm1.Input
            For i = LBound(inbyte) To UBound(inbyte)
                If Len(Hex(inbyte(i))) = 1 Then
                    buffer = buffer + "0" + Hex(inbyte(i)) + Chr(32)
                Else
                    buffer = buffer + Hex(inbyte(i)) + Chr(32)
                End If
            Next i
            
            lenbuffer = Len(Replace(Trim(buffer), " ", ""))
            Select Case lenbuffer
                Case Is = 14
                    If CByte("&H" & (Mid(Replace(buffer, " ", ""), 1, 2))) = &HAA _
                       And "&H" & (Mid(Replace(buffer, " ", ""), 14 - 4 + 1, 2)) = &H42 _
                       And "&H" & (Mid(Replace(buffer, " ", ""), 14 - 2 + 1, 2)) = &H42 Then
                           If "&H" & (Mid(Replace(buffer, " ", ""), 14 - 12 + 1, 2)) = &H55 Then
                               If "&H" & (Mid(Replace(buffer, " ", ""), 14 - 10 + 1, 2)) = &H43 _
                                    And "&H" & (Mid(Replace(buffer, " ", ""), 14 - 8 + 1, 2)) = &H47 _
                                    And "&H" & (Mid(Replace(buffer, " ", ""), 14 - 6 + 1, 2)) = &H8A Then
                                    Text2.Text = "????"
                               End If
                           End If
                    End If
             End Select
    End Select
End Sub
zdingyun 2013-07-29
  • 打赏
  • 举报
回复
将buffer定义为窗体变量
Option Explicit
    Dim buffer As String
    
Private Sub Form_Load()
    Dim i As Integer
    If MSComm1.PortOpen = True Then
        MSComm1.PortOpen = False
    End If
    MSComm1.CommPort = 1
    MSComm1.Settings = "115200,n,8,1"
    MSComm1.InputLen = 0
    MSComm1.InputMode = comInputModeBinary
    MSComm1.InBufferSize = 1024
    MSComm1.OutBufferSize = 2048
    MSComm1.InBufferCount = 0
    MSComm1.OutBufferCount = 0
    MSComm1.SThreshold = 0
    MSComm1.RThreshold = 1
    MSComm1.PortOpen = True '打开COM口
    Command2.Enabled = False
End Sub
 
Private Sub MSComm1_OnComm()
    Dim inbyte() As Byte
    Dim lenbuffer As Integer
    Dim i As Integer
    Select Case MSComm1.CommEvent
        Case comEvReceive
            inbyte = MSComm1.Input
            For i = LBound(inbyte) To UBound(inbyte)
                If Len(Hex(inbyte(i))) = 1 Then
                    buffer = buffer + "0" + Hex(inbyte(i)) + Chr(32)
                Else
                    buffer = buffer + Hex(inbyte(i)) + Chr(32)
                End If
            Next i
            Text1 = buffer
            lenbuffer = Len(Replace(Trim(buffer), " ", ""))
            Select Case lenbuffer
                Case Is = 14
                    If CByte("&H" & (Mid(Replace(buffer, " ", ""), 1, 2))) = &HAA _
                       And "&H" & (Mid(Replace(buffer, " ", ""), 14 - 4 + 1, 2)) = &H42 _
                       And "&H" & (Mid(Replace(buffer, " ", ""), 14 - 2 + 1, 2)) = &H42 Then
                           If "&H" & (Mid(Replace(buffer, " ", ""), 14 - 12 + 1, 2)) = &H55 Then
                               If "&H" & (Mid(Replace(buffer, " ", ""), 14 - 10 + 1, 2)) = &H43 _
                                    And "&H" & (Mid(Replace(buffer, " ", ""), 14 - 8 + 1, 2)) = &H47 _
                                    And "&H" & (Mid(Replace(buffer, " ", ""), 14 - 6 + 1, 2)) = &H8A Then
                                    Text2.Text = "????"
                               End If
                           End If
                    End If
             End Select
    End Select
End Sub
tornado_dai 2013-07-29
  • 打赏
  • 举报
回复
因为上位机需要处理的数据只有7字节和20字节两种情况,我修改成
 
Select Case MSComm1.CommEvent         
Case comEvReceive        
 '***********************************         
Do While MSComm1.InBufferCount <> 7 And MSComm1.InBufferCount <> 20            
   DoEvents            
    Loop         
inbyte = MSComm1.Input         
'***********************************
接收的数据就是完整的了,还是不解原来的为何数据丢失。。。如上修改后如果期间发送了不是7字节和20字节的数据,发送字节小于7,则是缓存数据到了7字节或20字节,才能收到数据,若是发送了一个2字节内容,再发送那个20字节的回复数据,就不能正确响应case事件了,显示收不到数据,因为加上缓存数据的2字节,此时为22字节,就直接不能处理了
tornado_dai 2013-07-29
  • 打赏
  • 举报
回复
因为上位机需要处理的数据只有7字节和20字节两种情况,我修改成[code=vb][/ Select Case MSComm1.CommEvent Case comEvReceive '*********************************** Do While MSComm1.InBufferCount <> 7 And MSComm1.InBufferCount <> 20 DoEvents Loop inbyte = MSComm1.Input '***********************************]接收的数据就是完整的了,还是不解原来的为何数据丢失。。。如上修改后如果期间发送了不是7字节和20字节的数据,发送字节小于7,则是缓存数据到了7字节或20字节,才能收到数据,若是发送了一个2字节内容,再发送那个20字节的回复数据,就不能正确响应case事件了,显示收不到数据,因为加上缓存数据的2字节,此时为22字节,就直接不能处理了
tornado_dai 2013-07-29
  • 打赏
  • 举报
回复
把[code=vb][/ For i = LBound(inbyte) To UBound(inbyte) If Len(Hex(inbyte(i))) = 1 Then buffer = buffer + "0" + Hex(inbyte(i)) + Chr(32) Else buffer = buffer + Hex(inbyte(i)) + Chr(32) End If Next i] 改为[code=vb][/buffer = buffer + Hex(inbyte(i)) + Chr(32)] 还是丢失数据,发送aa 55 43 47 8a 42 42 只收到55 43 47 8a 42 42
yachong 2013-07-29
  • 打赏
  • 举报
回复
前几年有一次写一个小串口程序放到客户某机器上面接收一小段数据,也是楼主这个现象 后来丢掉MSCOMM,网上找了一段api读写串口的代码就没问题了。再后来也没再研究这个事。
of123 2013-07-29
  • 打赏
  • 举报
回复
应该是上位机代码屏蔽掉了一些数据。 如果你要调试接收是否正常,把你那些判断条件什么的都注释掉,把接收到的所有数据都显示出来。
tornado_dai 2013-07-29
  • 打赏
  • 举报
回复
引用 9 楼 myjian 的回复:
把收到的数据全部输出来显示,再与发送的数据相比较,看看到底是完全不对,还是顺序不对,还是有缺失,再进行下一步排查.
显示出来看了,数据有缺失, 发送aa 43 47 8A 42 42的时候,只显示43 47 8A 42 42;发送aa 55 89 00 31 32 30 35 33 36 32 31 33 36 37 39 0f 05 42 42的时候,只显示89 00 31 32 30 35 33 36 32 31 33 36 37 39 0f 05 42 42,是哪里出问题了?
嗷嗷叫的老马 2013-07-29
  • 打赏
  • 举报
回复
另外,代码写在[code=vb]这里[/code <---此处还有一个]
嗷嗷叫的老马 2013-07-29
  • 打赏
  • 举报
回复
把收到的数据全部输出来显示,再与发送的数据相比较,看看到底是完全不对,还是顺序不对,还是有缺失,再进行下一步排查.
tornado_dai 2013-07-29
  • 打赏
  • 举报
回复
引用 7 楼 zdingyun 的回复:
[quote=引用 6 楼 daiyuankai 的回复:] 不好意思,我表述有误,情况是收到数据不对,上位机不能正确响应,下位机每次发送的数据和上位机收到的数据不一致
请降低波特率调试代码,或许USB转RS232线质量有问题导致。[/quote] 确定USB转串口线没有问题,波特率调低试过,也是收到的数据不对,还有其他原因吗?是不是要判断一下数据有没有接收完成还是怎样的
zdingyun 2013-07-29
  • 打赏
  • 举报
回复
引用 6 楼 daiyuankai 的回复:
不好意思,我表述有误,情况是收到数据不对,上位机不能正确响应,下位机每次发送的数据和上位机收到的数据不一致
请降低波特率调试代码,或许USB转RS232线质量有问题导致。
加载更多回复(4)
001、VB串口通讯视频教程源码41个 002、Visual Basic串口通信工程开发实例导航随书源码7个 003、Visual Basic串口通信与测控应用技术实战详解 源代码(15个全) 004、GE PLC串口通讯,VB编制,读取内存单元 005、PC机与51单片机之间的串口通讯,VB编的,分PC和单片机两部分 006、VB6的串口通信程序,还有crc校验 007、VB Modbus RTU源码,其中协议部分已生成DLL,可直接调用 008、VB.net开发的串口调试程序 009、VB.net实现串口编程,希望大家有用 010、VB版串口调试程序,包含VB源码及安装文件,适合调试串口 011、VB编程RS232串口控制DA数模转换 012、VB编程实现的串口调试工具源码 013、VB编写的RS232串口通信测试程序,以txt格式接受,可定义发送字符 014、VB编写的SouthStar串口测试与51串口烧写器V1.0版 015、VB编写的串口调试助手1.0的源码 016、VB编写的串口短信发送程序,需要数据线支持 017、VB编写的串口通信程序,实现多机通信 018、VB编写的串口通信程序,主要用于上位机与下位机间的通信 019、VB编写的串口通信程序界面参考网上的程序较简单 020、VB编写的串口通讯界面,主要面向51单片机的串口通信 021、VB编写的单片机和PC串口通信的调试程序 022、VB编写的仿真实电子琴操作界面,包含与FPGA串口通信的功能 023、VB串口API通讯,附带BAS文件全部源码,实现与饭卡读卡器通讯 024、VB串口编程,关于上位机的应用,特别适合初级学习VB的学员 025、VB串口编程调试精灵源码 026、VB串口编程实现完整的多费率电表读数软件 027、VB串口程序,,是一个串口使用例程,对初学者有用,特别是工控类的 028、VB串口传输文本,实现2台PC间的通信,类似简单的聊天工具 029、VB串口的一个电子称的项目 030、VB串口调试程序,用于通过串口控制松下空调测试 031、VB串口调试程序及源码 032、VB串口调试软件源代码,可以参考修改为其它通讯程序 033、VB串口调试软件源文件 034、VB串口控制步进电机程序完整源码 035、VB串口通信 6路10位AD转换数据采集源程序 036、VB串口通信,API串口通信模块源码 037、VB串口通信,适用简单,适合初学者 038、VB串口通信操作界面,进行数据采集,画实时曲线 039、VB串口通信程序,可以读取串口并显示保存数据,且能显示数据曲线 040、VB串口通信的源码,学习的好资料 041、VB串口通信调试器的源码程序 042、VB串口通信设计视频演示源码 043、VB串口通信示例 044、VB串口通信数据源码 045、VB串口通信之串口接收程序 046、VB串口通讯测试源代码,有文本和图形两种端口数据观察方式 047、VB串口通讯程序,用来跟单片机通讯 048、VB串口通讯代码(部分) 049、VB串口通讯的参考源程序 050、VB串口通讯实例 高精度电压表(24bit) VB源程序 051、vb串口通讯示例 052、VB串口与伺服电机DSP2407通讯完整代码源程序 053、VB串口源码,动力电池检测数据采集,内含电导巡检模块通讯报文,可,读写,保存,备份数据 054、VB串口字节通信程序,包括:1字节发送子程序,n字节接收子程序 055、VB串行口通信测试示例 056、VB串行通信试验程序 057、VB的MODEM通信源代码,智能化水电远端数据读取系统 058、VB的串口源程序,包括串口的配置界面,接收功能和发送功能 059、VB访问串口,并读取电子秤上显示的数据 060、VB和西门子S7-300 PLC串口通讯程序能实现读写功能 061、VB检测串口工作状态 062、VB简单的串口短信收发功能,使用短信猫测试通过 063、VB开发串口通信,关于生物医学工程专业的血氧饱和度的设计 064、VB开发串口通信软件,利用按钮控件控制高清晰数字展示台 065、VB开发的RS232串口图像处理器驱动(摄像头驱动) 066、VB开发的串口通信源码 067、VB开发的串口与三菱FX PLC通讯源码 068、VB控制串口232通讯,对飞利浦M1卡内数据进行处理,支持密码修改等 069、VB利用Mscomm控件编写的通讯终端,可做串口通讯编程参考示例 070、VB平台单片机与PC机串口通信的PC端程序。小巧易用,功能丰富 071、VB嵌入式串口通讯波形分析显示软件 072、VB实现串口调试LED信息显示屏设备主要代码 073、VB实现串口调试工具的完整源码 074、vb实现串口通信 文件传送系统,用vb以及mscomm控件实现 075、VB实现串口通信,发送命令从而接收相应数据 076、VB使用mscom控件实现PC机与单片机串口通信 077、VB通过COM串口读取条形码设备 078、VB通过串口控制单片机读写24C02源代码 079、VB通讯程序,连接串口可在电脑显示来电号码 080、VB下的串口发短信程序,可选择端口,设置短信中心号码 081、VB写的串口通信,发送和接收实例 082、VB写的串口通信分析程序源码 083、VB写的串口通讯,通过串口对单片机进行控制 084、VB写的串口通讯软件,简单易学,适合初学者 085、VB写的通过串口与考勤机连接通讯的程序 086、vb用控件的写的串口程序,是vb的经典之作 087、VB与USB转串口的通讯完整程序,有详细说明,不需要安装驱动 088、vb与串口通信的关于回路测试的小程序很实用 089、vb语言开发的串口通信,可实现拨号传送文件等 090、VB中串口事件处理函数的示例 091、VB中的串口通讯,串口通讯作为一种古老而又灵活的通讯方式,被广泛地应用 092、VB自动枚举系统串口加摄象头图象采集,坐标系变换 093、Visual Basic2005与自动化系统监控(串并行控制)光盘
本套监控系统非纯软件构成,而是由软硬件结合的监控程序和控制电路两部分组成,虽然控制电路只有几个元件,线路也简单的很,但一点动手能力没有的朋友就请不要下载了。   很多单位都拥有自己的服务器,像WEB服务器、FTP服务器、SMTP服务器等需要24时接入Internet。大家知道,局域网中的计算机基本上都是通过宽带路由器接入Internet,服务器的日常管理与维护通常都是通过Internet进行远程管理,所以,一旦路由器出现异常或停止工作,远程管理将完全无法实现,此时只能靠人工操作来重启路由器了。针对这种现状,作者开发了这套《宽带路由器死机监控系统》,以实现路由器死机后的自动重启。   监控程序布置在服务器(以下称监控计算机)上,并随监控计算机启动自动运行,每隔一定时间Ping路由器IP一次,如果Ping得通,只将信息记入日志不发出动作指令,如果Ping不通,监控程序会将Ping路由器的时间间隔缩短,当Ping路由器连续失败一定次数后,认定是路由器死机,监控程序通过计算机串口发出重启路由器的动作指令。   控制电路的核心元件是一只继电器,继电器的触点开关与路由器的电源开关(或复位启动开关)相连。控制电路还与监控计算机的串口相连。当控制电路接收到来自监控计算机串口输出的动作指令后继电器动作,控制路由器自动重启。   常用路由器有硬路由和由计算机构建的软路由两种。对于前者,控制电路的继电器使用的是常闭触点开关;对于后者,控制电路的继电器使用的是常开触点开关。   1.使用硬路由(要求控制电路的继电器常闭触点开关与硬路由电源开关串联):控制电路接收到动作指令,继电器吸合30秒后释放,相当于硬路由断电30秒后重新加电启动。   2.使用软路由(要求控制电路的继电器常开触点开关与软路由计算机电源开关并联):控制电路接收到动作指令,继电器吸合6秒后释放,延时30秒,继电器再次吸合2秒后释放,相当于计算机电源开关被按下六秒后强制关机,30秒后,计算机电源开关被再次按下,计算机重新启动。   3.使用软路由(要求控制电路的继电器常开触点开关与软路由计算机复位启动开关并联):控制电路接收到动作指令,继电器吸合2秒后释放,相当于计算机复位启动开关被按下2秒后松开,计算机重新启动。   控制电路和接口连接图请见压缩包中的说明文件(注:说明文件所附电路图误将“串口母关”标注成了“串口公头”,CSDN不允许更新上传的压缩包文件,所以请下载的朋友注意修正)。   监控软件用VB编写,只有一个主程序文件,将COMON.EXE复制到监控计算机上,并将其添加到Windows的计划任务中,设置为“计算机启动时运行”,以Administrator的权限运行。如果程序启动时出错,请用下面方法注册MSCOMM32.OCX文件:   将MSCOMM32.OCX拷贝到Windows\system32文件夹中,在“开始-运行”框内输入“REGSVR32.EXE MSCOMM32.OCX”确定。

1,451

社区成员

发帖
与我相关
我的任务
社区描述
VB 控件
社区管理员
  • 控件
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧